@switchbot/homebridge-switchbot 5.0.0-beta.5 → 5.0.0-beta.50

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.
Files changed (262) hide show
  1. package/CHANGELOG.md +15 -0
  2. package/README.md +45 -3
  3. package/config.schema.json +866 -13754
  4. package/dist/devices-hap/airpurifier.d.ts.map +1 -1
  5. package/dist/devices-hap/airpurifier.js +12 -6
  6. package/dist/devices-hap/airpurifier.js.map +1 -1
  7. package/dist/devices-hap/blindtilt.js +3 -3
  8. package/dist/devices-hap/bot.d.ts.map +1 -1
  9. package/dist/devices-hap/bot.js +16 -5
  10. package/dist/devices-hap/bot.js.map +1 -1
  11. package/dist/devices-hap/ceilinglight.d.ts.map +1 -1
  12. package/dist/devices-hap/ceilinglight.js +13 -7
  13. package/dist/devices-hap/ceilinglight.js.map +1 -1
  14. package/dist/devices-hap/colorbulb.d.ts.map +1 -1
  15. package/dist/devices-hap/colorbulb.js +49 -9
  16. package/dist/devices-hap/colorbulb.js.map +1 -1
  17. package/dist/devices-hap/contact.js +3 -3
  18. package/dist/devices-hap/curtain.js +2 -2
  19. package/dist/devices-hap/curtain.js.map +1 -1
  20. package/dist/devices-hap/device.d.ts +18 -8
  21. package/dist/devices-hap/device.d.ts.map +1 -1
  22. package/dist/devices-hap/device.js +141 -69
  23. package/dist/devices-hap/device.js.map +1 -1
  24. package/dist/devices-hap/fan.d.ts.map +1 -1
  25. package/dist/devices-hap/fan.js +12 -6
  26. package/dist/devices-hap/fan.js.map +1 -1
  27. package/dist/devices-hap/hub.d.ts.map +1 -1
  28. package/dist/devices-hap/hub.js +6 -5
  29. package/dist/devices-hap/hub.js.map +1 -1
  30. package/dist/devices-hap/humidifier.d.ts +5 -0
  31. package/dist/devices-hap/humidifier.d.ts.map +1 -1
  32. package/dist/devices-hap/humidifier.js +92 -4
  33. package/dist/devices-hap/humidifier.js.map +1 -1
  34. package/dist/devices-hap/iosensor.d.ts.map +1 -1
  35. package/dist/devices-hap/iosensor.js +36 -21
  36. package/dist/devices-hap/iosensor.js.map +1 -1
  37. package/dist/devices-hap/lightstrip.d.ts.map +1 -1
  38. package/dist/devices-hap/lightstrip.js +38 -8
  39. package/dist/devices-hap/lightstrip.js.map +1 -1
  40. package/dist/devices-hap/lock.d.ts.map +1 -1
  41. package/dist/devices-hap/lock.js +14 -6
  42. package/dist/devices-hap/lock.js.map +1 -1
  43. package/dist/devices-hap/meter.d.ts.map +1 -1
  44. package/dist/devices-hap/meter.js +6 -5
  45. package/dist/devices-hap/meter.js.map +1 -1
  46. package/dist/devices-hap/meterplus.d.ts.map +1 -1
  47. package/dist/devices-hap/meterplus.js +6 -5
  48. package/dist/devices-hap/meterplus.js.map +1 -1
  49. package/dist/devices-hap/meterpro.d.ts.map +1 -1
  50. package/dist/devices-hap/meterpro.js +7 -6
  51. package/dist/devices-hap/meterpro.js.map +1 -1
  52. package/dist/devices-hap/motion.js +3 -3
  53. package/dist/devices-hap/plug.d.ts.map +1 -1
  54. package/dist/devices-hap/plug.js +11 -6
  55. package/dist/devices-hap/plug.js.map +1 -1
  56. package/dist/devices-hap/relayswitch.js +3 -3
  57. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +1 -1
  58. package/dist/devices-hap/robotvacuumcleaner.js +13 -6
  59. package/dist/devices-hap/robotvacuumcleaner.js.map +1 -1
  60. package/dist/devices-hap/waterdetector.js +3 -3
  61. package/dist/devices-matter/BaseMatterAccessory.d.ts +27 -0
  62. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +1 -1
  63. package/dist/devices-matter/BaseMatterAccessory.js +169 -5
  64. package/dist/devices-matter/BaseMatterAccessory.js.map +1 -1
  65. package/dist/devices-matter/ColorLightAccessory.d.ts.map +1 -1
  66. package/dist/devices-matter/ColorLightAccessory.js +12 -12
  67. package/dist/devices-matter/ColorLightAccessory.js.map +1 -1
  68. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +1 -1
  69. package/dist/devices-matter/ColorTemperatureLightAccessory.js +5 -7
  70. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +1 -1
  71. package/dist/devices-matter/DimmableLightAccessory.js +9 -9
  72. package/dist/devices-matter/DimmableLightAccessory.js.map +1 -1
  73. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +1 -1
  74. package/dist/devices-matter/ExtendedColorLightAccessory.js +14 -15
  75. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +1 -1
  76. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +1 -1
  77. package/dist/devices-matter/OnOffLightAccessory.js +8 -16
  78. package/dist/devices-matter/OnOffLightAccessory.js.map +1 -1
  79. package/dist/devices-matter/OnOffOutletAccessory.d.ts +2 -0
  80. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +1 -1
  81. package/dist/devices-matter/OnOffOutletAccessory.js +10 -7
  82. package/dist/devices-matter/OnOffOutletAccessory.js.map +1 -1
  83. package/dist/devices-matter/OnOffSwitchAccessory.js +2 -2
  84. package/dist/devices-matter/OnOffSwitchAccessory.js.map +1 -1
  85. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +11 -2
  86. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +1 -1
  87. package/dist/devices-matter/RoboticVacuumAccessory.js +26 -17
  88. package/dist/devices-matter/RoboticVacuumAccessory.js.map +1 -1
  89. package/dist/homebridge-ui/public/index.html +64 -2
  90. package/dist/homebridge-ui/server.js +77 -9
  91. package/dist/homebridge-ui/server.js.map +1 -1
  92. package/dist/index.d.ts.map +1 -1
  93. package/dist/index.js +4 -7
  94. package/dist/index.js.map +1 -1
  95. package/dist/irdevice/irdevice.d.ts +11 -10
  96. package/dist/irdevice/irdevice.d.ts.map +1 -1
  97. package/dist/irdevice/irdevice.js +76 -35
  98. package/dist/irdevice/irdevice.js.map +1 -1
  99. package/dist/platform-hap.d.ts +26 -15
  100. package/dist/platform-hap.d.ts.map +1 -1
  101. package/dist/platform-hap.js +330 -153
  102. package/dist/platform-hap.js.map +1 -1
  103. package/dist/platform-matter.d.ts +98 -6
  104. package/dist/platform-matter.d.ts.map +1 -1
  105. package/dist/platform-matter.js +1884 -254
  106. package/dist/platform-matter.js.map +1 -1
  107. package/dist/settings.d.ts +58 -7
  108. package/dist/settings.d.ts.map +1 -1
  109. package/dist/settings.js.map +1 -1
  110. package/dist/test/apiRequestTracker.test.d.ts +2 -0
  111. package/dist/test/apiRequestTracker.test.d.ts.map +1 -0
  112. package/dist/test/apiRequestTracker.test.js +392 -0
  113. package/dist/test/apiRequestTracker.test.js.map +1 -0
  114. package/dist/test/hap/device-webhook-context.test.d.ts +2 -0
  115. package/dist/test/hap/device-webhook-context.test.d.ts.map +1 -0
  116. package/dist/test/hap/device-webhook-context.test.js +128 -0
  117. package/dist/test/hap/device-webhook-context.test.js.map +1 -0
  118. package/dist/test/hap/platform-hap.logging.test.d.ts +2 -0
  119. package/dist/test/hap/platform-hap.logging.test.d.ts.map +1 -0
  120. package/dist/test/hap/platform-hap.logging.test.js +33 -0
  121. package/dist/test/hap/platform-hap.logging.test.js.map +1 -0
  122. package/dist/test/hap/platform-hap.test.d.ts +2 -0
  123. package/dist/test/hap/platform-hap.test.d.ts.map +1 -0
  124. package/dist/test/hap/platform-hap.test.js +62 -0
  125. package/dist/test/hap/platform-hap.test.js.map +1 -0
  126. package/dist/test/helpers/platform-fixtures.d.ts +9 -0
  127. package/dist/test/helpers/platform-fixtures.d.ts.map +1 -0
  128. package/dist/test/helpers/platform-fixtures.js +30 -0
  129. package/dist/test/helpers/platform-fixtures.js.map +1 -0
  130. package/dist/test/homebridge-ui/server.test.d.ts +2 -0
  131. package/dist/test/homebridge-ui/server.test.d.ts.map +1 -0
  132. package/dist/test/homebridge-ui/server.test.js +445 -0
  133. package/dist/test/homebridge-ui/server.test.js.map +1 -0
  134. package/dist/{index.test.d.ts.map → test/index.test.d.ts.map} +1 -1
  135. package/dist/test/index.test.js +19 -0
  136. package/dist/test/index.test.js.map +1 -0
  137. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts +2 -0
  138. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts.map +1 -0
  139. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js +71 -0
  140. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js.map +1 -0
  141. package/dist/test/matter/platform-matter.additional.test.d.ts +2 -0
  142. package/dist/test/matter/platform-matter.additional.test.d.ts.map +1 -0
  143. package/dist/test/matter/platform-matter.additional.test.js +35 -0
  144. package/dist/test/matter/platform-matter.additional.test.js.map +1 -0
  145. package/dist/test/matter/platform-matter.bleparse.test.d.ts +2 -0
  146. package/dist/test/matter/platform-matter.bleparse.test.d.ts.map +1 -0
  147. package/dist/test/matter/platform-matter.bleparse.test.js +43 -0
  148. package/dist/test/matter/platform-matter.bleparse.test.js.map +1 -0
  149. package/dist/test/matter/platform-matter.cleanup.test.d.ts +2 -0
  150. package/dist/test/matter/platform-matter.cleanup.test.d.ts.map +1 -0
  151. package/dist/test/matter/platform-matter.cleanup.test.js +70 -0
  152. package/dist/test/matter/platform-matter.cleanup.test.js.map +1 -0
  153. package/dist/test/matter/platform-matter.keepstale.test.d.ts +2 -0
  154. package/dist/test/matter/platform-matter.keepstale.test.d.ts.map +1 -0
  155. package/dist/test/matter/platform-matter.keepstale.test.js +27 -0
  156. package/dist/test/matter/platform-matter.keepstale.test.js.map +1 -0
  157. package/dist/test/matter/platform-matter.logging.test.d.ts +2 -0
  158. package/dist/test/matter/platform-matter.logging.test.d.ts.map +1 -0
  159. package/dist/test/matter/platform-matter.logging.test.js +29 -0
  160. package/dist/test/matter/platform-matter.logging.test.js.map +1 -0
  161. package/dist/test/matter/platform-matter.mapping.test.d.ts +2 -0
  162. package/dist/test/matter/platform-matter.mapping.test.d.ts.map +1 -0
  163. package/dist/test/matter/platform-matter.mapping.test.js +43 -0
  164. package/dist/test/matter/platform-matter.mapping.test.js.map +1 -0
  165. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts +2 -0
  166. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts.map +1 -0
  167. package/dist/test/matter/platform-matter.openapi-mapping.test.js +84 -0
  168. package/dist/test/matter/platform-matter.openapi-mapping.test.js.map +1 -0
  169. package/dist/test/matter/platform-matter.test.d.ts +2 -0
  170. package/dist/test/matter/platform-matter.test.d.ts.map +1 -0
  171. package/dist/test/matter/platform-matter.test.js +117 -0
  172. package/dist/test/matter/platform-matter.test.js.map +1 -0
  173. package/dist/test/matter/platform-matter.unregister.test.d.ts +2 -0
  174. package/dist/test/matter/platform-matter.unregister.test.d.ts.map +1 -0
  175. package/dist/test/matter/platform-matter.unregister.test.js +30 -0
  176. package/dist/test/matter/platform-matter.unregister.test.js.map +1 -0
  177. package/dist/test/matter/platform-matter.webhook.test.d.ts +2 -0
  178. package/dist/test/matter/platform-matter.webhook.test.d.ts.map +1 -0
  179. package/dist/test/matter/platform-matter.webhook.test.js +46 -0
  180. package/dist/test/matter/platform-matter.webhook.test.js.map +1 -0
  181. package/dist/test/utils.test.d.ts +2 -0
  182. package/dist/test/utils.test.d.ts.map +1 -0
  183. package/dist/test/utils.test.js +95 -0
  184. package/dist/test/utils.test.js.map +1 -0
  185. package/dist/test/verifyconfig.test.d.ts.map +1 -0
  186. package/dist/{verifyconfig.test.js → test/verifyconfig.test.js} +2 -2
  187. package/dist/test/verifyconfig.test.js.map +1 -0
  188. package/dist/utils.d.ts +204 -3
  189. package/dist/utils.d.ts.map +1 -1
  190. package/dist/utils.js +713 -33
  191. package/dist/utils.js.map +1 -1
  192. package/docs/assets/highlight.css +14 -0
  193. package/docs/assets/main.js +2 -2
  194. package/docs/index.html +31 -2
  195. package/docs/variables/default.html +1 -1
  196. package/package.json +15 -15
  197. package/src/devices-hap/airpurifier.ts +11 -6
  198. package/src/devices-hap/blindtilt.ts +3 -3
  199. package/src/devices-hap/bot.ts +15 -5
  200. package/src/devices-hap/ceilinglight.ts +12 -7
  201. package/src/devices-hap/colorbulb.ts +46 -10
  202. package/src/devices-hap/contact.ts +3 -3
  203. package/src/devices-hap/curtain.ts +2 -2
  204. package/src/devices-hap/device.ts +149 -70
  205. package/src/devices-hap/fan.ts +11 -6
  206. package/src/devices-hap/hub.ts +6 -5
  207. package/src/devices-hap/humidifier.ts +97 -4
  208. package/src/devices-hap/iosensor.ts +36 -21
  209. package/src/devices-hap/lightstrip.ts +35 -8
  210. package/src/devices-hap/lock.ts +13 -6
  211. package/src/devices-hap/meter.ts +6 -5
  212. package/src/devices-hap/meterplus.ts +6 -5
  213. package/src/devices-hap/meterpro.ts +7 -6
  214. package/src/devices-hap/motion.ts +3 -3
  215. package/src/devices-hap/plug.ts +10 -6
  216. package/src/devices-hap/relayswitch.ts +3 -3
  217. package/src/devices-hap/robotvacuumcleaner.ts +12 -6
  218. package/src/devices-hap/waterdetector.ts +3 -3
  219. package/src/devices-matter/BaseMatterAccessory.ts +176 -5
  220. package/src/devices-matter/ColorLightAccessory.ts +12 -12
  221. package/src/devices-matter/ColorTemperatureLightAccessory.ts +5 -7
  222. package/src/devices-matter/DimmableLightAccessory.ts +9 -9
  223. package/src/devices-matter/ExtendedColorLightAccessory.ts +14 -15
  224. package/src/devices-matter/OnOffLightAccessory.ts +8 -16
  225. package/src/devices-matter/OnOffOutletAccessory.ts +12 -7
  226. package/src/devices-matter/OnOffSwitchAccessory.ts +2 -2
  227. package/src/devices-matter/RoboticVacuumAccessory.ts +27 -17
  228. package/src/homebridge-ui/public/index.html +64 -2
  229. package/src/homebridge-ui/server.ts +80 -9
  230. package/src/index.ts +4 -7
  231. package/src/irdevice/irdevice.ts +74 -35
  232. package/src/platform-hap.ts +362 -169
  233. package/src/platform-matter.ts +1937 -257
  234. package/src/settings.ts +62 -3
  235. package/src/test/apiRequestTracker.test.ts +417 -0
  236. package/src/test/hap/device-webhook-context.test.ts +136 -0
  237. package/src/test/hap/platform-hap.logging.test.ts +36 -0
  238. package/src/test/hap/platform-hap.test.ts +70 -0
  239. package/src/test/helpers/platform-fixtures.ts +33 -0
  240. package/src/test/homebridge-ui/server.test.ts +486 -0
  241. package/src/test/index.test.ts +24 -0
  242. package/src/test/matter/devices-matter/baseMatterAccessory.test.ts +88 -0
  243. package/src/test/matter/platform-matter.additional.test.ts +44 -0
  244. package/src/test/matter/platform-matter.bleparse.test.ts +47 -0
  245. package/src/test/matter/platform-matter.cleanup.test.ts +86 -0
  246. package/src/test/matter/platform-matter.keepstale.test.ts +37 -0
  247. package/src/test/matter/platform-matter.logging.test.ts +33 -0
  248. package/src/test/matter/platform-matter.mapping.test.ts +57 -0
  249. package/src/test/matter/platform-matter.openapi-mapping.test.ts +109 -0
  250. package/src/test/matter/platform-matter.test.ts +144 -0
  251. package/src/test/matter/platform-matter.unregister.test.ts +39 -0
  252. package/src/test/matter/platform-matter.webhook.test.ts +54 -0
  253. package/src/test/utils.test.ts +96 -0
  254. package/src/{verifyconfig.test.ts → test/verifyconfig.test.ts} +12 -11
  255. package/src/utils.ts +777 -36
  256. package/dist/index.test.js +0 -14
  257. package/dist/index.test.js.map +0 -1
  258. package/dist/verifyconfig.test.d.ts.map +0 -1
  259. package/dist/verifyconfig.test.js.map +0 -1
  260. package/src/index.test.ts +0 -19
  261. /package/dist/{index.test.d.ts → test/index.test.d.ts} +0 -0
  262. /package/dist/{verifyconfig.test.d.ts → test/verifyconfig.test.d.ts} +0 -0
@@ -0,0 +1,88 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { BaseMatterAccessory } from '../../../devices-matter/BaseMatterAccessory.js'
4
+
5
+ // Minimal concrete subclass for testing
6
+ class TestAccessory extends BaseMatterAccessory {
7
+ constructor(api: any, log: any, opts?: any) {
8
+ super(api, log, {
9
+ uuid: opts?.uuid ?? api.matter.uuid.generate('test'),
10
+ displayName: opts?.displayName ?? 'Test',
11
+ deviceType: opts?.deviceType ?? ('OnOffLight' as any),
12
+ serialNumber: opts?.serialNumber ?? 'TEST-1',
13
+ manufacturer: opts?.manufacturer ?? 'TestCo',
14
+ model: opts?.model ?? 'T-1',
15
+ firmwareRevision: opts?.firmwareRevision ?? '1.0',
16
+ hardwareRevision: opts?.hardwareRevision ?? '1.0',
17
+ context: opts?.context ?? {},
18
+ clusters: opts?.clusters,
19
+ handlers: opts?.handlers,
20
+ })
21
+ }
22
+ }
23
+
24
+ describe('baseMatterAccessory helpers', () => {
25
+ it('sendOnCommand uses OpenAPI helper and updates state', async () => {
26
+ const update = vi.fn()
27
+ const sendOpenAPI = vi.fn(async () => ({ response: {}, statusCode: 200 }))
28
+ const api: any = { matter: { clusterNames: { OnOff: 'onOff', LevelControl: 'level', ColorControl: 'color' }, uuid: { generate: () => 'uuid-1' }, updateAccessoryState: update } }
29
+ const log: any = { info: vi.fn(), debug: vi.fn(), error: vi.fn(), warn: vi.fn() }
30
+
31
+ const ctx = { deviceId: 'DEV-1', sendOpenAPI, connectionType: 'OpenAPI' }
32
+ const acc = new TestAccessory(api, log, { context: ctx })
33
+
34
+ await acc.sendOnCommand()
35
+
36
+ expect(sendOpenAPI).toHaveBeenCalledWith('turnOn', 'default')
37
+ expect(update).toHaveBeenCalledWith(acc.uuid, 'onOff', { onOff: true })
38
+ })
39
+
40
+ it('sendOnCommand uses BLE helper when connectionType is BLE', async () => {
41
+ const update = vi.fn()
42
+ const sendBLE = vi.fn(async () => true)
43
+ const api: any = { matter: { clusterNames: { OnOff: 'onOff', LevelControl: 'level', ColorControl: 'color' }, uuid: { generate: () => 'uuid-2' }, updateAccessoryState: update } }
44
+ const log: any = { info: vi.fn(), debug: vi.fn(), error: vi.fn(), warn: vi.fn() }
45
+
46
+ const ctx = { deviceId: 'DEV-2', sendBLE, connectionType: 'BLE' }
47
+ const acc = new TestAccessory(api, log, { context: ctx })
48
+
49
+ await acc.sendOnCommand()
50
+
51
+ expect(sendBLE).toHaveBeenCalledWith('turnOn')
52
+ expect(update).toHaveBeenCalledWith(acc.uuid, 'onOff', { onOff: true })
53
+ })
54
+
55
+ it('sendSetBrightness sends percent and updates LevelControl', async () => {
56
+ const update = vi.fn()
57
+ const sendOpenAPI = vi.fn(async () => ({ response: {}, statusCode: 200 }))
58
+ const api: any = { matter: { clusterNames: { OnOff: 'onOff', LevelControl: 'level', ColorControl: 'color' }, uuid: { generate: () => 'uuid-3' }, updateAccessoryState: update } }
59
+ const log: any = { info: vi.fn(), debug: vi.fn(), error: vi.fn(), warn: vi.fn() }
60
+
61
+ const ctx = { deviceId: 'DEV-3', sendOpenAPI, connectionType: 'OpenAPI' }
62
+ const acc = new TestAccessory(api, log, { context: ctx })
63
+
64
+ await acc.sendSetBrightness(50)
65
+
66
+ expect(sendOpenAPI).toHaveBeenCalledWith('setBrightness', '50')
67
+ const expectedLevel = Math.round((50 / 100) * 254)
68
+ expect(update).toHaveBeenCalledWith(acc.uuid, 'level', { currentLevel: expectedLevel })
69
+ })
70
+
71
+ it('sendSetColor sends RGB and updates ColorControl', async () => {
72
+ const update = vi.fn()
73
+ const sendOpenAPI = vi.fn(async () => ({ response: {}, statusCode: 200 }))
74
+ const api: any = { matter: { clusterNames: { OnOff: 'onOff', LevelControl: 'level', ColorControl: 'color' }, uuid: { generate: () => 'uuid-4' }, updateAccessoryState: update } }
75
+ const log: any = { info: vi.fn(), debug: vi.fn(), error: vi.fn(), warn: vi.fn() }
76
+
77
+ const ctx = { deviceId: 'DEV-4', sendOpenAPI, connectionType: 'OpenAPI' }
78
+ const acc = new TestAccessory(api, log, { context: ctx })
79
+
80
+ await acc.sendSetColor(255, 128, 0)
81
+
82
+ expect(sendOpenAPI).toHaveBeenCalledWith('setColor', '255:128:0')
83
+ expect(update).toHaveBeenCalled()
84
+ // Ensure we updated ColorControl cluster with numeric attributes
85
+ const calledWith = (update.mock.calls[0] || update.mock.calls[1])
86
+ expect(calledWith[1] === 'color' || calledWith[1] === api.matter.clusterNames.ColorControl).toBeTruthy()
87
+ })
88
+ })
@@ -0,0 +1,44 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
5
+
6
+ describe('additional platform-matter mapping tests', () => {
7
+ it('parses OpenAPI color strings and triggers an update', async () => {
8
+ const updateAccessoryState = vi.fn()
9
+ const api: any = makeApiStub({ updateAccessoryState })
10
+ const log = makeLogStub()
11
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
12
+
13
+ // Some OpenAPI statuses provide color as comma/colon separated or hex
14
+ await (platform as any).applyStatusToAccessory('uuid-col', { deviceId: 'DEV-COL' } as any, { color: '255:128:64' })
15
+ await (platform as any).applyStatusToAccessory('uuid-col', { deviceId: 'DEV-COL' } as any, { color: '255,128,64' })
16
+ await (platform as any).applyStatusToAccessory('uuid-col', { deviceId: 'DEV-COL' } as any, { color: '#FF8040' })
17
+
18
+ expect(updateAccessoryState).toHaveBeenCalled()
19
+ })
20
+
21
+ it('maps MeterPro CO2 values to Matter updates', async () => {
22
+ const updateAccessoryState = vi.fn()
23
+ const api: any = makeApiStub({ updateAccessoryState })
24
+ const log = makeLogStub()
25
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
26
+
27
+ await (platform as any).applyStatusToAccessory('uuid-co2', { deviceId: 'DEV-METERPRO' } as any, { co2: 420 })
28
+
29
+ expect(updateAccessoryState).toHaveBeenCalled()
30
+ })
31
+
32
+ it('handles curtain position synonyms (position / slidePosition)', async () => {
33
+ const updateAccessoryState = vi.fn()
34
+ const api: any = makeApiStub({ updateAccessoryState })
35
+ const log = makeLogStub()
36
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
37
+
38
+ // Provide status with position and slidePosition synonyms
39
+ await (platform as any).applyStatusToAccessory('uuid-cur', { deviceId: 'DEV-CUR', deviceType: 'Curtain' } as any, { position: 30 })
40
+ await (platform as any).applyStatusToAccessory('uuid-cur', { deviceId: 'DEV-CUR', deviceType: 'Curtain' } as any, { slidePosition: 70 })
41
+
42
+ expect(updateAccessoryState).toHaveBeenCalled()
43
+ })
44
+ })
@@ -0,0 +1,47 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
5
+
6
+ describe('platform-matter BLE advertisement parser', () => {
7
+ it('parses extended serviceData fields correctly', () => {
8
+ const updateAccessoryState = vi.fn()
9
+ const api: any = makeApiStub({ updateAccessoryState })
10
+
11
+ const log: any = makeLogStub()
12
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
13
+
14
+ const dev: any = { deviceId: 'DEV-TEST' }
15
+ const sd = {
16
+ temp: 22.3,
17
+ humid: 45,
18
+ pm25: 12,
19
+ pm10: 34,
20
+ voc: 120,
21
+ co2: 420,
22
+ motion: 1,
23
+ open: 0,
24
+ leak: 0,
25
+ position: 30,
26
+ fanSpeed: 75,
27
+ battery: 88,
28
+ rgb: '255:128:64',
29
+ }
30
+
31
+ const parsed = (platform as any).parseAdvertisementForDevice(dev, sd)
32
+ expect(parsed).toBeTruthy()
33
+ expect(parsed.temperature).toBeCloseTo(22.3)
34
+ expect(parsed.humidity).toBe(45)
35
+ expect(parsed.pm25).toBe(12)
36
+ expect(parsed.pm10).toBe(34)
37
+ expect(parsed.voc).toBe(120)
38
+ expect(parsed.co2).toBe(420)
39
+ expect(parsed.motion).toBeTruthy()
40
+ expect(parsed.contact).toBeDefined()
41
+ expect(parsed.leak).toBeFalsy()
42
+ expect(parsed.position).toBe(30)
43
+ expect(parsed.fanSpeed).toBe(75)
44
+ expect(parsed.battery).toBe(88)
45
+ expect(parsed.color).toEqual({ r: 255, g: 128, b: 64 })
46
+ })
47
+ })
@@ -0,0 +1,86 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { formatDeviceIdAsMac } from '../../utils.js'
5
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
6
+
7
+ describe('platform-matter lifecycle cleanup', () => {
8
+ it('clearDeviceResources removes timers, instances and BLE handler entries', async () => {
9
+ // Setup stubbed API and logs
10
+ const handlers: Record<string, (...args: any[]) => any> = {}
11
+ const api: any = makeApiStub({ registerPlatformAccessories: vi.fn(), unregisterPlatformAccessories: vi.fn(), clusterNames: { OnOff: 'OnOff' } })
12
+ api._handlers = handlers
13
+ // keep api.on as a single-statement arrow to satisfy lint
14
+ api.on = (ev: string, fn: (...args: any[]) => any) => (handlers[ev] = fn)
15
+
16
+ const log = makeLogStub()
17
+
18
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
19
+
20
+ // Insert a fake timer, accessory instance, and BLE handler
21
+ const deviceId = 'AA:BB:CC:11:22:33'
22
+ const nid = (platform as any).normalizeDeviceId(deviceId)
23
+
24
+ const timer = setInterval(() => {}, 100000)
25
+ ;(platform as any).refreshTimers.set(nid, timer)
26
+ ;(platform as any).accessoryInstances.set(nid, { dummy: true })
27
+ ;(platform as any).bleEventHandler[deviceId.toLowerCase()] = () => {}
28
+
29
+ // Ensure they exist prior
30
+ expect((platform as any).refreshTimers.get(nid)).toBeDefined()
31
+ expect((platform as any).accessoryInstances.get(nid)).toBeDefined()
32
+ expect((platform as any).bleEventHandler[deviceId.toLowerCase()]).toBeDefined()
33
+
34
+ // Call the private helper
35
+ ;(platform as any).clearDeviceResources(deviceId)
36
+
37
+ // Now they should be removed
38
+ expect((platform as any).refreshTimers.get(nid)).toBeUndefined()
39
+ expect((platform as any).accessoryInstances.get(nid)).toBeUndefined()
40
+ expect((platform as any).bleEventHandler[deviceId.toLowerCase()]).toBeUndefined()
41
+ })
42
+
43
+ it('shutdown handler clears all timers and handlers when invoked', async () => {
44
+ // Setup stubbed API and logs
45
+ const handlers: Record<string, (...args: any[]) => any> = {}
46
+ const api: any = makeApiStub({ registerPlatformAccessories: vi.fn(), unregisterPlatformAccessories: vi.fn(), clusterNames: { OnOff: 'OnOff' } })
47
+ api._handlers = handlers
48
+ // keep api.on as a single-statement arrow to satisfy lint
49
+ api.on = (ev: string, fn: (...args: any[]) => any) => (handlers[ev] = fn)
50
+
51
+ const log = makeLogStub()
52
+
53
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
54
+
55
+ // Add two timers to the platform (using normalized ids)
56
+ const ids = ['devA', 'devB']
57
+ for (const id of ids) {
58
+ const nid = (platform as any).normalizeDeviceId(id)
59
+ const t = setInterval(() => {}, 100000)
60
+ ;(platform as any).refreshTimers.set(nid, t)
61
+ ;(platform as any).accessoryInstances.set(nid, { dummy: true })
62
+ try {
63
+ const mac = formatDeviceIdAsMac(id).toLowerCase()
64
+ ;(platform as any).bleEventHandler[mac] = () => {}
65
+ } catch {
66
+ // ignore formatting errors in this test
67
+ }
68
+ }
69
+
70
+ // Invoke didFinishLaunching to ensure the platform registered its shutdown handler
71
+ await Promise.resolve(api._handlers.didFinishLaunching?.())
72
+
73
+ // Shutdown handler should now be registered on api._handlers.shutdown
74
+ expect(typeof api._handlers.shutdown).toBe('function')
75
+
76
+ // Call the shutdown handler
77
+ await Promise.resolve(api._handlers.shutdown())
78
+
79
+ // All refresh timers should be cleared
80
+ for (const id of ids) {
81
+ const nid = (platform as any).normalizeDeviceId(id)
82
+ expect((platform as any).refreshTimers.get(nid)).toBeUndefined()
83
+ expect((platform as any).accessoryInstances.get(nid)).toBeUndefined()
84
+ }
85
+ })
86
+ })
@@ -0,0 +1,37 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
5
+
6
+ describe('keepStaleAccessories config flag behavior', () => {
7
+ it('keeps previously-registered accessories when options.keepStaleAccessories=true', async () => {
8
+ const unregister = vi.fn()
9
+ const register = vi.fn()
10
+ const api: any = makeApiStub({ unregisterPlatformAccessories: unregister, registerPlatformAccessories: register })
11
+
12
+ const log: any = makeLogStub()
13
+
14
+ // Create platform with keepStaleAccessories = true
15
+ const platform = new SwitchBotMatterPlatform(log as any, { options: { keepStaleAccessories: true } } as any, api)
16
+
17
+ // Seed discovered devices (one device)
18
+ ;(platform as any).discoveredDevices = [{ deviceId: 'DEV1', deviceType: 'Plug', deviceName: 'Device 1' }]
19
+
20
+ // Insert a stale accessory into matterAccessories
21
+ const staleUuid = api.matter.uuid.generate('stale-device')
22
+ const staleAcc: any = { uuid: staleUuid, displayName: 'Stale', context: { deviceId: 'STALE_DEVICE' } }
23
+ ;(platform as any).matterAccessories.set(staleUuid, staleAcc)
24
+
25
+ // Mock createAccessoryFromDevice to return a simple accessory for DEV1
26
+ vi.spyOn(platform as any, 'createAccessoryFromDevice').mockResolvedValue({ displayName: 'Device 1', uuid: 'uuid-DEV1', context: { deviceId: 'DEV1' } } as any)
27
+
28
+ // Run registration which includes stale-removal logic
29
+ await (platform as any).registerMatterAccessories()
30
+
31
+ // Expect unregister NOT called (we kept stale accessory)
32
+ expect(unregister).not.toHaveBeenCalled()
33
+
34
+ // Stale accessory should still be present in matterAccessories
35
+ expect((platform as any).matterAccessories.get(staleUuid)).toBeDefined()
36
+ })
37
+ })
@@ -0,0 +1,33 @@
1
+ import { describe, expect, it } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
5
+
6
+ /**
7
+ * Verifies that debug logger formats arguments so that accessory displayName
8
+ * appears in the output for cached accessory load logs.
9
+ */
10
+ describe('platform-matter logging', () => {
11
+ it('prints accessory name when loading cached Matter accessory', async () => {
12
+ const api: any = makeApiStub()
13
+ const log: any = makeLogStub()
14
+
15
+ const platform = new SwitchBotMatterPlatform(log as any, {
16
+ name: 'SwitchBot',
17
+ credentials: {},
18
+ options: { logging: 'debug' },
19
+ } as any, api)
20
+
21
+ // Simulate Homebridge restoring a cached Matter accessory
22
+ const acc: any = { uuid: 'uuid-TEST', displayName: 'Test Device', context: { deviceId: 'DEV1' } }
23
+ ;(platform as any).configureMatterAccessory(acc)
24
+ // debugLog is async and gated; yield to allow the logger to run
25
+ await new Promise(resolve => setTimeout(resolve, 0))
26
+
27
+ // In 'debug' mode, debugLog uses log.info with a [DEBUG] prefix
28
+ // Ensure one of the info calls contains the message with the device name
29
+ const calls = (log.info as any).mock.calls as Array<string[]>
30
+ const hasLine = calls.some(args => String(args[0]).includes('Loading cached Matter accessory: Test Device'))
31
+ expect(hasLine).toBe(true)
32
+ })
33
+ })
@@ -0,0 +1,57 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
5
+
6
+ describe('platform-matter mapping helper', () => {
7
+ it('prefers accessory instance update methods for battery and falls back to api.matter.updateAccessoryState', async () => {
8
+ const updateAccessoryState = vi.fn()
9
+ const api: any = makeApiStub({ updateAccessoryState, clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl', PowerSource: 'powerSource' } })
10
+
11
+ const log: any = makeLogStub()
12
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
13
+
14
+ // Create a fake accessory instance with updateBatteryPercentage
15
+ const deviceId = 'DEV123'
16
+ const nid = (platform as any).normalizeDeviceId(deviceId)
17
+ const fakeInstance = { updateBatteryPercentage: vi.fn() }
18
+ ;(platform as any).accessoryInstances.set(nid, fakeInstance)
19
+
20
+ // Call the private helper with different battery field names
21
+ await (platform as any).applyStatusToAccessory('uuid-test', { deviceId } as any, { batteryPercentage: 55 })
22
+ expect(fakeInstance.updateBatteryPercentage).toHaveBeenCalled()
23
+
24
+ // Remove instance to force fallback
25
+ ;(platform as any).accessoryInstances.delete(nid)
26
+ await (platform as any).applyStatusToAccessory('uuid-test', { deviceId } as any, { battery: 30 })
27
+ expect(updateAccessoryState).toHaveBeenCalled()
28
+ })
29
+
30
+ it('handles temperature and humidity synonyms', async () => {
31
+ const updateAccessoryState = vi.fn()
32
+ const api: any = makeApiStub({ updateAccessoryState })
33
+
34
+ const log: any = makeLogStub()
35
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
36
+
37
+ const deviceId = 'DEV-TEMP'
38
+ await (platform as any).applyStatusToAccessory('uuid-temp', { deviceId } as any, { temp: 21.5, humid: 48 })
39
+
40
+ // Expect updateAccessoryState to have been called for temperature and humidity
41
+ expect(updateAccessoryState).toHaveBeenCalled()
42
+ })
43
+
44
+ it('applies VOC and PM10 mappings when present', async () => {
45
+ const updateAccessoryState = vi.fn()
46
+ const api: any = makeApiStub({ updateAccessoryState })
47
+
48
+ const log: any = makeLogStub()
49
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
50
+
51
+ const deviceId = 'DEV-AQ'
52
+ await (platform as any).applyStatusToAccessory('uuid-aq', { deviceId } as any, { voc: 123, pm10: 56 })
53
+
54
+ // Expect updateAccessoryState (or safeUpdate fallback) to have been called for voc and pm10
55
+ expect(updateAccessoryState).toHaveBeenCalled()
56
+ })
57
+ })
@@ -0,0 +1,109 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
5
+
6
+ describe('platform-matter OpenAPI -> Matter mapping', () => {
7
+ it('maps light OpenAPI status (on/off, brightness, color, battery) to accessory instance helpers', async () => {
8
+ const api: any = makeApiStub()
9
+ const log = makeLogStub()
10
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
11
+
12
+ const deviceId = 'LIGHT-1'
13
+ const nid = (platform as any).normalizeDeviceId(deviceId)
14
+
15
+ const instance = {
16
+ updateOnOffState: vi.fn(),
17
+ updateBrightness: vi.fn(),
18
+ updateHueSaturation: vi.fn(),
19
+ updateBatteryPercentage: vi.fn(),
20
+ }
21
+ ;(platform as any).accessoryInstances.set(nid, instance)
22
+
23
+ const status = { power: true, brightness: 80, color: '255:128:64', battery: 92 }
24
+ await (platform as any).applyStatusToAccessory('uuid-light', { deviceId } as any, status)
25
+
26
+ expect(instance.updateOnOffState).toHaveBeenCalled()
27
+ expect(instance.updateBrightness).toHaveBeenCalled()
28
+ expect(instance.updateHueSaturation).toHaveBeenCalled()
29
+ expect(instance.updateBatteryPercentage).toHaveBeenCalled()
30
+ })
31
+
32
+ it('maps MeterPro OpenAPI status (temp, humid, co2, pm25, voc) to accessory instance helpers', async () => {
33
+ const api: any = makeApiStub()
34
+ const log = makeLogStub()
35
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
36
+
37
+ const deviceId = 'METERPRO-1'
38
+ const nid = (platform as any).normalizeDeviceId(deviceId)
39
+
40
+ const instance = {
41
+ updateTemperature: vi.fn(),
42
+ updateHumidity: vi.fn(),
43
+ updateCO2: vi.fn(),
44
+ updatePM25: vi.fn(),
45
+ updateVOC: vi.fn(),
46
+ }
47
+ ;(platform as any).accessoryInstances.set(nid, instance)
48
+
49
+ const status = { temp: 21.4, humid: 45, co2: 410, pm25: 12, voc: 85 }
50
+ await (platform as any).applyStatusToAccessory('uuid-meter', { deviceId } as any, status)
51
+
52
+ expect(instance.updateTemperature).toHaveBeenCalled()
53
+ expect(instance.updateHumidity).toHaveBeenCalled()
54
+ expect(instance.updateCO2).toHaveBeenCalled()
55
+ expect(instance.updatePM25).toHaveBeenCalled()
56
+ expect(instance.updateVOC).toHaveBeenCalled()
57
+ })
58
+
59
+ it('maps lock OpenAPI status to lock helper', async () => {
60
+ const api: any = makeApiStub()
61
+ const log = makeLogStub()
62
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
63
+
64
+ const deviceId = 'LOCK-1'
65
+ const nid = (platform as any).normalizeDeviceId(deviceId)
66
+
67
+ const instance = { updateLockState: vi.fn() }
68
+ ;(platform as any).accessoryInstances.set(nid, instance)
69
+
70
+ const status = { lock: true }
71
+ await (platform as any).applyStatusToAccessory('uuid-lock', { deviceId } as any, status)
72
+
73
+ expect(instance.updateLockState).toHaveBeenCalled()
74
+ })
75
+
76
+ it('maps curtain position synonyms to lift position helper', async () => {
77
+ const api: any = makeApiStub()
78
+ const log = makeLogStub()
79
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
80
+
81
+ const deviceId = 'CURTAIN-1'
82
+ const nid = (platform as any).normalizeDeviceId(deviceId)
83
+
84
+ const instance = { updateLiftPosition: vi.fn() }
85
+ ;(platform as any).accessoryInstances.set(nid, instance)
86
+
87
+ await (platform as any).applyStatusToAccessory('uuid-cur', { deviceId, deviceType: 'Curtain' } as any, { position: 25 })
88
+ await (platform as any).applyStatusToAccessory('uuid-cur', { deviceId, deviceType: 'Curtain' } as any, { slidePosition: 75 })
89
+
90
+ expect(instance.updateLiftPosition).toHaveBeenCalled()
91
+ })
92
+
93
+ it('maps robot vacuum run state / on to updateRunMode or updateOperationalState', async () => {
94
+ const api: any = makeApiStub()
95
+ const log = makeLogStub()
96
+ const platform = new SwitchBotMatterPlatform(log as any, {} as any, api)
97
+
98
+ const deviceId = 'RVC-1'
99
+ const nid = (platform as any).normalizeDeviceId(deviceId)
100
+
101
+ const instance: any = { updateRunMode: vi.fn(), updateOperationalState: vi.fn() }
102
+ ;(platform as any).accessoryInstances.set(nid, instance)
103
+
104
+ await (platform as any).applyStatusToAccessory('uuid-rvc', { deviceId, deviceType: 'K10+' } as any, { runState: 'cleaning', power: true })
105
+
106
+ const calls = (instance.updateRunMode.mock?.calls?.length || 0) + (instance.updateOperationalState.mock?.calls?.length || 0)
107
+ expect(calls).toBeGreaterThan(0)
108
+ })
109
+ })
@@ -0,0 +1,144 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotMatterPlatform } from '../../platform-matter.js'
4
+ import { PLATFORM_NAME, PLUGIN_NAME } from '../../settings.js'
5
+ import { makeApiStub, makeLogStub } from '../helpers/platform-fixtures.js'
6
+
7
+ describe('platform-matter discovered devices', () => {
8
+ it('uses discovered devices and registers them (platform + robotic)', async () => {
9
+ const mockRegister = vi.fn()
10
+ const api: any = makeApiStub({
11
+ registerPlatformAccessories: mockRegister,
12
+ clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl' },
13
+ deviceTypes: { RoboticVacuumCleaner: 'rvc' },
14
+ })
15
+
16
+ const log: any = makeLogStub()
17
+
18
+ const config: any = {}
19
+
20
+ const platform = new SwitchBotMatterPlatform(log, config, api)
21
+
22
+ // Provide discovered devices
23
+ ;(platform as any).discoveredDevices = [
24
+ { deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' },
25
+ { deviceId: 'vac1', deviceName: 'Vac', deviceType: 'K10+' },
26
+ ] as any
27
+
28
+ // Stub accessory creation to avoid instantiating full device classes
29
+ ;(platform as any).createAccessoryFromDevice = async (dev: any) => ({ displayName: dev.deviceName, uuid: api.matter.uuid.generate(dev.deviceId) } as any)
30
+
31
+ await (platform as any).registerMatterAccessories()
32
+
33
+ // Ensure registerPlatformAccessories was called at least once
34
+ expect(mockRegister).toHaveBeenCalled()
35
+
36
+ // Sum total accessories registered across all calls
37
+ const totalRegistered = mockRegister.mock.calls.reduce((sum: number, call: any) => sum + ((call[2] && call[2].length) || 0), 0)
38
+ expect(totalRegistered).toBe(2)
39
+
40
+ // Verify PLUGIN_NAME and PLATFORM_NAME were used in the calls
41
+ for (const call of mockRegister.mock.calls) {
42
+ expect(call[0]).toBe(PLUGIN_NAME)
43
+ expect(call[1]).toBe(PLATFORM_NAME)
44
+ }
45
+ })
46
+
47
+ it('applies per-device config overrides when deviceId matches discovered device', async () => {
48
+ const mockRegister = vi.fn()
49
+
50
+ const api: any = {
51
+ matter: {
52
+ uuid: { generate: (s: string) => `uuid-${s}` },
53
+ registerPlatformAccessories: mockRegister,
54
+ clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl' },
55
+ deviceTypes: { RoboticVacuumCleaner: 'rvc' },
56
+ },
57
+ isMatterAvailable: () => true,
58
+ isMatterEnabled: () => true,
59
+ on: () => {},
60
+ }
61
+
62
+ const log: any = { info: vi.fn(), warn: vi.fn(), debug: vi.fn(), error: vi.fn() }
63
+
64
+ const config: any = {
65
+ options: {
66
+ devices: [{ deviceId: 'dev1', configDeviceName: 'Configured Lamp', logging: 'debug' }],
67
+ },
68
+ }
69
+
70
+ const platform = new SwitchBotMatterPlatform(log, config, api)
71
+
72
+ ;(platform as any).discoveredDevices = [{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' }] as any
73
+
74
+ const seen: any[] = []
75
+ ;(platform as any).createAccessoryFromDevice = async (dev: any) => {
76
+ seen.push(dev)
77
+ return {
78
+ displayName: dev.deviceName ?? dev.configDeviceName,
79
+ uuid: api.matter.uuid.generate(dev.deviceId),
80
+ } as any
81
+ }
82
+
83
+ await (platform as any).registerMatterAccessories()
84
+
85
+ // createAccessoryFromDevice should have been called once with merged config
86
+ expect(seen.length).toBe(1)
87
+ expect(seen[0].deviceId).toBe('dev1')
88
+ // per-device config values should be present on the merged device
89
+ expect(seen[0].configDeviceName).toBe('Configured Lamp')
90
+ expect(seen[0].logging).toBe('debug')
91
+ })
92
+
93
+ it('ignores config-only devices by default but includes them when allowConfigOnlyDevices=true', async () => {
94
+ const mockRegister = vi.fn()
95
+
96
+ const api: any = {
97
+ matter: {
98
+ uuid: { generate: (s: string) => `uuid-${s}` },
99
+ registerPlatformAccessories: mockRegister,
100
+ clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl' },
101
+ deviceTypes: { RoboticVacuumCleaner: 'rvc' },
102
+ },
103
+ isMatterAvailable: () => true,
104
+ isMatterEnabled: () => true,
105
+ on: () => {},
106
+ }
107
+
108
+ const log: any = { info: vi.fn(), warn: vi.fn(), debug: vi.fn(), error: vi.fn() }
109
+
110
+ // config-only device (cfg1) and one discovered device (dev1)
111
+ const cfgOnly = { deviceId: 'cfg1', configDeviceName: 'OnlyInConfig', deviceType: 'Plug' }
112
+ const configDefault: any = { options: { devices: [cfgOnly] } }
113
+
114
+ const platformDefault = new SwitchBotMatterPlatform(log, configDefault, api)
115
+ ;(platformDefault as any).discoveredDevices = [{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' }] as any
116
+ const seenDefault: any[] = []
117
+ ;(platformDefault as any).createAccessoryFromDevice = async (dev: any) => {
118
+ seenDefault.push(dev)
119
+ return {
120
+ displayName: dev.deviceName ?? dev.configDeviceName,
121
+ uuid: api.matter.uuid.generate(dev.deviceId),
122
+ } as any
123
+ }
124
+ await (platformDefault as any).registerMatterAccessories()
125
+ // Should only see discovered device (cfg1 ignored)
126
+ expect(seenDefault.find((d: any) => d.deviceId === 'cfg1')).toBeUndefined()
127
+
128
+ // Now opt-in to include config-only devices
129
+ const configOptIn: any = { options: { devices: [cfgOnly], allowConfigOnlyDevices: true } }
130
+ const platformOptIn = new SwitchBotMatterPlatform(log, configOptIn, api)
131
+ ;(platformOptIn as any).discoveredDevices = [{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' }] as any
132
+ const seenOptIn: any[] = []
133
+ ;(platformOptIn as any).createAccessoryFromDevice = async (dev: any) => {
134
+ seenOptIn.push(dev)
135
+ return {
136
+ displayName: dev.deviceName ?? dev.configDeviceName,
137
+ uuid: api.matter.uuid.generate(dev.deviceId),
138
+ } as any
139
+ }
140
+ await (platformOptIn as any).registerMatterAccessories()
141
+ // Should include the config-only device now
142
+ expect(seenOptIn.find((d: any) => d.deviceId === 'cfg1')).toBeDefined()
143
+ })
144
+ })