@switchbot/homebridge-switchbot 5.0.0-beta.70 → 5.0.0-beta.72

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 (534) hide show
  1. package/.github/ISSUE_TEMPLATE/e2e-verification.md +36 -0
  2. package/.github/workflows/ci.yml +61 -0
  3. package/.github/workflows/manual-e2e.yml +103 -0
  4. package/CHANGELOG.md +20 -0
  5. package/E2E-VERIFICATION.md +121 -0
  6. package/MIGRATION.md +44 -0
  7. package/README.md +11 -0
  8. package/config.schema.json +99 -1940
  9. package/dist/deviceFactory.d.ts +13 -0
  10. package/dist/deviceFactory.d.ts.map +1 -0
  11. package/dist/deviceFactory.js +81 -0
  12. package/dist/deviceFactory.js.map +1 -0
  13. package/dist/devices/deviceBase.d.ts +50 -0
  14. package/dist/devices/deviceBase.d.ts.map +1 -0
  15. package/dist/devices/deviceBase.js +119 -0
  16. package/dist/devices/deviceBase.js.map +1 -0
  17. package/dist/devices/genericDevice.d.ts +283 -0
  18. package/dist/devices/genericDevice.d.ts.map +1 -0
  19. package/dist/devices/genericDevice.js +1035 -0
  20. package/dist/devices/genericDevice.js.map +1 -0
  21. package/dist/homebridge-ui/public/index.html +72 -440
  22. package/dist/homebridge-ui/server.d.ts +3 -1
  23. package/dist/homebridge-ui/server.d.ts.map +1 -1
  24. package/dist/homebridge-ui/server.js +47 -10
  25. package/dist/homebridge-ui/server.js.map +1 -1
  26. package/dist/index.d.ts.map +1 -1
  27. package/dist/index.js +1 -3
  28. package/dist/index.js.map +1 -1
  29. package/dist/platform.d.ts +27 -0
  30. package/dist/platform.d.ts.map +1 -0
  31. package/dist/platform.js +404 -0
  32. package/dist/platform.js.map +1 -0
  33. package/dist/settings.d.ts +10 -317
  34. package/dist/settings.d.ts.map +1 -1
  35. package/dist/settings.js +5 -30
  36. package/dist/settings.js.map +1 -1
  37. package/dist/switchbotClient.d.ts +32 -0
  38. package/dist/switchbotClient.d.ts.map +1 -0
  39. package/dist/switchbotClient.js +259 -0
  40. package/dist/switchbotClient.js.map +1 -0
  41. package/dist/utils.d.ts +36 -248
  42. package/dist/utils.d.ts.map +1 -1
  43. package/dist/utils.js +38 -1367
  44. package/dist/utils.js.map +1 -1
  45. package/docs/assets/icons.js +1 -1
  46. package/docs/assets/icons.svg +1 -1
  47. package/docs/assets/style.css +3 -3
  48. package/docs/index.html +50 -15
  49. package/docs/variables/default.html +1 -1
  50. package/package.json +19 -18
  51. package/scripts/e2e/README.md +25 -0
  52. package/scripts/e2e/curtain-e2e.sh +70 -0
  53. package/scripts/e2e/fan-e2e.sh +75 -0
  54. package/scripts/e2e/light-advanced-e2e.sh +97 -0
  55. package/scripts/e2e/light-e2e.sh +75 -0
  56. package/scripts/e2e/list-accessories.sh +19 -0
  57. package/scripts/e2e/lock-e2e.sh +65 -0
  58. package/scripts/generate-matter-maps.js +60 -0
  59. package/scripts/run-e2e-local.sh +14 -0
  60. package/src/deviceFactory.ts +122 -0
  61. package/src/devices/deviceBase.ts +141 -0
  62. package/src/devices/genericDevice.ts +965 -0
  63. package/src/homebridge-ui/public/index.html +72 -440
  64. package/src/homebridge-ui/server.ts +52 -10
  65. package/src/index.ts +1 -3
  66. package/src/platform.ts +395 -0
  67. package/src/settings.ts +12 -352
  68. package/src/switchbotClient.ts +266 -0
  69. package/src/utils.ts +47 -1456
  70. package/test/accessory-restore.spec.ts +73 -0
  71. package/test/device-mapping.spec.ts +37 -0
  72. package/test/deviceFactory.spec.ts +18 -0
  73. package/test/e2e/run-e2e.spec.ts +50 -0
  74. package/test/fan-swing.spec.ts +29 -0
  75. package/test/helpers/matter-harness.ts +53 -0
  76. package/test/lock-users.spec.ts +44 -0
  77. package/test/matter-childbridge.spec.ts +55 -0
  78. package/test/matter-descriptors.spec.ts +97 -0
  79. package/test/matter-device-state.spec.ts +101 -0
  80. package/test/matter-integration.spec.ts +70 -0
  81. package/test/platform.integration.spec.ts +55 -0
  82. package/test/switchbot-client-debounce.spec.ts +131 -0
  83. package/test/switchbot-client-openapi.spec.ts +56 -0
  84. package/test/switchbotClient.spec.ts +10 -0
  85. package/test/utils.spec.ts +20 -0
  86. package/vitest.config.ts +7 -0
  87. package/coverage/base.css +0 -224
  88. package/coverage/block-navigation.js +0 -87
  89. package/coverage/clover.xml +0 -15847
  90. package/coverage/coverage-final.json +0 -42
  91. package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
  92. package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
  93. package/coverage/docs/assets/dmt/index.html +0 -131
  94. package/coverage/docs/assets/hierarchy.js.html +0 -85
  95. package/coverage/docs/assets/icons.js.html +0 -136
  96. package/coverage/docs/assets/index.html +0 -146
  97. package/coverage/docs/assets/main.js.html +0 -265
  98. package/coverage/favicon.png +0 -0
  99. package/coverage/index.html +0 -191
  100. package/coverage/prettify.css +0 -1
  101. package/coverage/prettify.js +0 -2
  102. package/coverage/sort-arrow-sprite.png +0 -0
  103. package/coverage/sorter.js +0 -196
  104. package/coverage/src/device/blindtilt.ts.html +0 -3238
  105. package/coverage/src/device/bot.ts.html +0 -2803
  106. package/coverage/src/device/ceilinglight.ts.html +0 -2338
  107. package/coverage/src/device/colorbulb.ts.html +0 -2824
  108. package/coverage/src/device/contact.ts.html +0 -1465
  109. package/coverage/src/device/curtain.ts.html +0 -2869
  110. package/coverage/src/device/device.ts.html +0 -2500
  111. package/coverage/src/device/fan.ts.html +0 -2242
  112. package/coverage/src/device/hub.ts.html +0 -1408
  113. package/coverage/src/device/humidifier.ts.html +0 -2116
  114. package/coverage/src/device/index.html +0 -416
  115. package/coverage/src/device/iosensor.ts.html +0 -1375
  116. package/coverage/src/device/lightstrip.ts.html +0 -2617
  117. package/coverage/src/device/lock.ts.html +0 -1963
  118. package/coverage/src/device/meter.ts.html +0 -1372
  119. package/coverage/src/device/meterplus.ts.html +0 -1384
  120. package/coverage/src/device/meterpro.ts.html +0 -1618
  121. package/coverage/src/device/motion.ts.html +0 -1264
  122. package/coverage/src/device/plug.ts.html +0 -1372
  123. package/coverage/src/device/relayswitch.ts.html +0 -2284
  124. package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
  125. package/coverage/src/device/waterdetector.ts.html +0 -1294
  126. package/coverage/src/homebridge-ui/index.html +0 -116
  127. package/coverage/src/homebridge-ui/server.ts.html +0 -229
  128. package/coverage/src/index.html +0 -161
  129. package/coverage/src/index.ts.html +0 -124
  130. package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
  131. package/coverage/src/irdevice/airpurifier.ts.html +0 -844
  132. package/coverage/src/irdevice/camera.ts.html +0 -475
  133. package/coverage/src/irdevice/fan.ts.html +0 -766
  134. package/coverage/src/irdevice/index.html +0 -251
  135. package/coverage/src/irdevice/irdevice.ts.html +0 -1117
  136. package/coverage/src/irdevice/light.ts.html +0 -826
  137. package/coverage/src/irdevice/other.ts.html +0 -2458
  138. package/coverage/src/irdevice/tv.ts.html +0 -1222
  139. package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
  140. package/coverage/src/irdevice/waterheater.ts.html +0 -469
  141. package/coverage/src/platform.ts.html +0 -8776
  142. package/coverage/src/settings.ts.html +0 -934
  143. package/coverage/src/utils.ts.html +0 -2092
  144. package/dist/devices-hap/airpurifier.d.ts +0 -54
  145. package/dist/devices-hap/airpurifier.d.ts.map +0 -1
  146. package/dist/devices-hap/airpurifier.js +0 -533
  147. package/dist/devices-hap/airpurifier.js.map +0 -1
  148. package/dist/devices-hap/blindtilt.d.ts +0 -90
  149. package/dist/devices-hap/blindtilt.d.ts.map +0 -1
  150. package/dist/devices-hap/blindtilt.js +0 -974
  151. package/dist/devices-hap/blindtilt.js.map +0 -1
  152. package/dist/devices-hap/bot.d.ts +0 -102
  153. package/dist/devices-hap/bot.d.ts.map +0 -1
  154. package/dist/devices-hap/bot.js +0 -822
  155. package/dist/devices-hap/bot.js.map +0 -1
  156. package/dist/devices-hap/ceilinglight.d.ts +0 -85
  157. package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
  158. package/dist/devices-hap/ceilinglight.js +0 -707
  159. package/dist/devices-hap/ceilinglight.js.map +0 -1
  160. package/dist/devices-hap/colorbulb.d.ts +0 -88
  161. package/dist/devices-hap/colorbulb.d.ts.map +0 -1
  162. package/dist/devices-hap/colorbulb.js +0 -921
  163. package/dist/devices-hap/colorbulb.js.map +0 -1
  164. package/dist/devices-hap/contact.d.ts +0 -44
  165. package/dist/devices-hap/contact.d.ts.map +0 -1
  166. package/dist/devices-hap/contact.js +0 -409
  167. package/dist/devices-hap/contact.js.map +0 -1
  168. package/dist/devices-hap/curtain.d.ts +0 -73
  169. package/dist/devices-hap/curtain.d.ts.map +0 -1
  170. package/dist/devices-hap/curtain.js +0 -869
  171. package/dist/devices-hap/curtain.js.map +0 -1
  172. package/dist/devices-hap/device.d.ts +0 -108
  173. package/dist/devices-hap/device.d.ts.map +0 -1
  174. package/dist/devices-hap/device.js +0 -821
  175. package/dist/devices-hap/device.js.map +0 -1
  176. package/dist/devices-hap/fan.d.ts +0 -69
  177. package/dist/devices-hap/fan.d.ts.map +0 -1
  178. package/dist/devices-hap/fan.js +0 -655
  179. package/dist/devices-hap/fan.js.map +0 -1
  180. package/dist/devices-hap/hub.d.ts +0 -37
  181. package/dist/devices-hap/hub.d.ts.map +0 -1
  182. package/dist/devices-hap/hub.js +0 -393
  183. package/dist/devices-hap/hub.js.map +0 -1
  184. package/dist/devices-hap/humidifier.d.ts +0 -73
  185. package/dist/devices-hap/humidifier.d.ts.map +0 -1
  186. package/dist/devices-hap/humidifier.js +0 -716
  187. package/dist/devices-hap/humidifier.js.map +0 -1
  188. package/dist/devices-hap/iosensor.d.ts +0 -42
  189. package/dist/devices-hap/iosensor.d.ts.map +0 -1
  190. package/dist/devices-hap/iosensor.js +0 -397
  191. package/dist/devices-hap/iosensor.js.map +0 -1
  192. package/dist/devices-hap/lightstrip.d.ts +0 -79
  193. package/dist/devices-hap/lightstrip.d.ts.map +0 -1
  194. package/dist/devices-hap/lightstrip.js +0 -827
  195. package/dist/devices-hap/lightstrip.js.map +0 -1
  196. package/dist/devices-hap/lock.d.ts +0 -53
  197. package/dist/devices-hap/lock.d.ts.map +0 -1
  198. package/dist/devices-hap/lock.js +0 -569
  199. package/dist/devices-hap/lock.js.map +0 -1
  200. package/dist/devices-hap/meter.d.ts +0 -37
  201. package/dist/devices-hap/meter.d.ts.map +0 -1
  202. package/dist/devices-hap/meter.js +0 -380
  203. package/dist/devices-hap/meter.js.map +0 -1
  204. package/dist/devices-hap/meterplus.d.ts +0 -42
  205. package/dist/devices-hap/meterplus.d.ts.map +0 -1
  206. package/dist/devices-hap/meterplus.js +0 -385
  207. package/dist/devices-hap/meterplus.js.map +0 -1
  208. package/dist/devices-hap/meterpro.d.ts +0 -43
  209. package/dist/devices-hap/meterpro.d.ts.map +0 -1
  210. package/dist/devices-hap/meterpro.js +0 -469
  211. package/dist/devices-hap/meterpro.js.map +0 -1
  212. package/dist/devices-hap/motion.d.ts +0 -42
  213. package/dist/devices-hap/motion.d.ts.map +0 -1
  214. package/dist/devices-hap/motion.js +0 -345
  215. package/dist/devices-hap/motion.js.map +0 -1
  216. package/dist/devices-hap/plug.d.ts +0 -49
  217. package/dist/devices-hap/plug.d.ts.map +0 -1
  218. package/dist/devices-hap/plug.js +0 -400
  219. package/dist/devices-hap/plug.js.map +0 -1
  220. package/dist/devices-hap/relayswitch.d.ts +0 -96
  221. package/dist/devices-hap/relayswitch.d.ts.map +0 -1
  222. package/dist/devices-hap/relayswitch.js +0 -642
  223. package/dist/devices-hap/relayswitch.js.map +0 -1
  224. package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
  225. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
  226. package/dist/devices-hap/robotvacuumcleaner.js +0 -530
  227. package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
  228. package/dist/devices-hap/waterdetector.d.ts +0 -41
  229. package/dist/devices-hap/waterdetector.d.ts.map +0 -1
  230. package/dist/devices-hap/waterdetector.js +0 -356
  231. package/dist/devices-hap/waterdetector.js.map +0 -1
  232. package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -90
  233. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
  234. package/dist/devices-matter/BaseMatterAccessory.js +0 -264
  235. package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
  236. package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
  237. package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
  238. package/dist/devices-matter/ColorLightAccessory.js +0 -95
  239. package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
  240. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
  241. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
  242. package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -76
  243. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
  244. package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
  245. package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
  246. package/dist/devices-matter/ContactSensorAccessory.js +0 -34
  247. package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
  248. package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
  249. package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
  250. package/dist/devices-matter/DimmableLightAccessory.js +0 -167
  251. package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
  252. package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
  253. package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
  254. package/dist/devices-matter/DoorLockAccessory.js +0 -50
  255. package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
  256. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
  257. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
  258. package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -106
  259. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
  260. package/dist/devices-matter/FanAccessory.d.ts +0 -16
  261. package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
  262. package/dist/devices-matter/FanAccessory.js +0 -81
  263. package/dist/devices-matter/FanAccessory.js.map +0 -1
  264. package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
  265. package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
  266. package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
  267. package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
  268. package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
  269. package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
  270. package/dist/devices-matter/LeakSensorAccessory.js +0 -33
  271. package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
  272. package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
  273. package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
  274. package/dist/devices-matter/LightSensorAccessory.js +0 -34
  275. package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
  276. package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
  277. package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
  278. package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
  279. package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
  280. package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
  281. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
  282. package/dist/devices-matter/OnOffLightAccessory.js +0 -110
  283. package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
  284. package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -14
  285. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
  286. package/dist/devices-matter/OnOffOutletAccessory.js +0 -43
  287. package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
  288. package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
  289. package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
  290. package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
  291. package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
  292. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -61
  293. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
  294. package/dist/devices-matter/RoboticVacuumAccessory.js +0 -544
  295. package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
  296. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
  297. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
  298. package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
  299. package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
  300. package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
  301. package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
  302. package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
  303. package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
  304. package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
  305. package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
  306. package/dist/devices-matter/ThermostatAccessory.js +0 -95
  307. package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
  308. package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
  309. package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
  310. package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
  311. package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
  312. package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
  313. package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
  314. package/dist/devices-matter/WindowBlindAccessory.js +0 -131
  315. package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
  316. package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
  317. package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
  318. package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
  319. package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
  320. package/dist/devices-matter/custom/index.d.ts +0 -8
  321. package/dist/devices-matter/custom/index.d.ts.map +0 -1
  322. package/dist/devices-matter/custom/index.js +0 -8
  323. package/dist/devices-matter/custom/index.js.map +0 -1
  324. package/dist/devices-matter/index.d.ts +0 -29
  325. package/dist/devices-matter/index.d.ts.map +0 -1
  326. package/dist/devices-matter/index.js +0 -28
  327. package/dist/devices-matter/index.js.map +0 -1
  328. package/dist/irdevice/airconditioner.d.ts +0 -61
  329. package/dist/irdevice/airconditioner.d.ts.map +0 -1
  330. package/dist/irdevice/airconditioner.js +0 -472
  331. package/dist/irdevice/airconditioner.js.map +0 -1
  332. package/dist/irdevice/airpurifier.d.ts +0 -50
  333. package/dist/irdevice/airpurifier.d.ts.map +0 -1
  334. package/dist/irdevice/airpurifier.js +0 -213
  335. package/dist/irdevice/airpurifier.js.map +0 -1
  336. package/dist/irdevice/camera.d.ts +0 -32
  337. package/dist/irdevice/camera.d.ts.map +0 -1
  338. package/dist/irdevice/camera.js +0 -107
  339. package/dist/irdevice/camera.js.map +0 -1
  340. package/dist/irdevice/fan.d.ts +0 -36
  341. package/dist/irdevice/fan.d.ts.map +0 -1
  342. package/dist/irdevice/fan.js +0 -200
  343. package/dist/irdevice/fan.js.map +0 -1
  344. package/dist/irdevice/irdevice.d.ts +0 -69
  345. package/dist/irdevice/irdevice.d.ts.map +0 -1
  346. package/dist/irdevice/irdevice.js +0 -339
  347. package/dist/irdevice/irdevice.js.map +0 -1
  348. package/dist/irdevice/light.d.ts +0 -36
  349. package/dist/irdevice/light.d.ts.map +0 -1
  350. package/dist/irdevice/light.js +0 -206
  351. package/dist/irdevice/light.js.map +0 -1
  352. package/dist/irdevice/other.d.ts +0 -57
  353. package/dist/irdevice/other.d.ts.map +0 -1
  354. package/dist/irdevice/other.js +0 -778
  355. package/dist/irdevice/other.js.map +0 -1
  356. package/dist/irdevice/tv.d.ts +0 -45
  357. package/dist/irdevice/tv.d.ts.map +0 -1
  358. package/dist/irdevice/tv.js +0 -327
  359. package/dist/irdevice/tv.js.map +0 -1
  360. package/dist/irdevice/vacuumcleaner.d.ts +0 -28
  361. package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
  362. package/dist/irdevice/vacuumcleaner.js +0 -104
  363. package/dist/irdevice/vacuumcleaner.js.map +0 -1
  364. package/dist/irdevice/waterheater.d.ts +0 -30
  365. package/dist/irdevice/waterheater.d.ts.map +0 -1
  366. package/dist/irdevice/waterheater.js +0 -105
  367. package/dist/irdevice/waterheater.js.map +0 -1
  368. package/dist/platform-hap.d.ts +0 -160
  369. package/dist/platform-hap.d.ts.map +0 -1
  370. package/dist/platform-hap.js +0 -3041
  371. package/dist/platform-hap.js.map +0 -1
  372. package/dist/platform-matter.d.ts +0 -188
  373. package/dist/platform-matter.d.ts.map +0 -1
  374. package/dist/platform-matter.js +0 -2545
  375. package/dist/platform-matter.js.map +0 -1
  376. package/dist/test/apiRequestTracker.test.d.ts +0 -2
  377. package/dist/test/apiRequestTracker.test.d.ts.map +0 -1
  378. package/dist/test/apiRequestTracker.test.js +0 -392
  379. package/dist/test/apiRequestTracker.test.js.map +0 -1
  380. package/dist/test/hap/device-webhook-context.test.d.ts +0 -2
  381. package/dist/test/hap/device-webhook-context.test.d.ts.map +0 -1
  382. package/dist/test/hap/device-webhook-context.test.js +0 -128
  383. package/dist/test/hap/device-webhook-context.test.js.map +0 -1
  384. package/dist/test/hap/platform-hap.logging.test.d.ts +0 -2
  385. package/dist/test/hap/platform-hap.logging.test.d.ts.map +0 -1
  386. package/dist/test/hap/platform-hap.logging.test.js +0 -33
  387. package/dist/test/hap/platform-hap.logging.test.js.map +0 -1
  388. package/dist/test/hap/platform-hap.test.d.ts +0 -2
  389. package/dist/test/hap/platform-hap.test.d.ts.map +0 -1
  390. package/dist/test/hap/platform-hap.test.js +0 -62
  391. package/dist/test/hap/platform-hap.test.js.map +0 -1
  392. package/dist/test/helpers/platform-fixtures.d.ts +0 -9
  393. package/dist/test/helpers/platform-fixtures.d.ts.map +0 -1
  394. package/dist/test/helpers/platform-fixtures.js +0 -30
  395. package/dist/test/helpers/platform-fixtures.js.map +0 -1
  396. package/dist/test/homebridge-ui/server.test.d.ts +0 -2
  397. package/dist/test/homebridge-ui/server.test.d.ts.map +0 -1
  398. package/dist/test/homebridge-ui/server.test.js +0 -445
  399. package/dist/test/homebridge-ui/server.test.js.map +0 -1
  400. package/dist/test/index.test.d.ts +0 -2
  401. package/dist/test/index.test.d.ts.map +0 -1
  402. package/dist/test/index.test.js +0 -19
  403. package/dist/test/index.test.js.map +0 -1
  404. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts +0 -2
  405. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts.map +0 -1
  406. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js +0 -71
  407. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js.map +0 -1
  408. package/dist/test/matter/platform-matter.additional.test.d.ts +0 -2
  409. package/dist/test/matter/platform-matter.additional.test.d.ts.map +0 -1
  410. package/dist/test/matter/platform-matter.additional.test.js +0 -35
  411. package/dist/test/matter/platform-matter.additional.test.js.map +0 -1
  412. package/dist/test/matter/platform-matter.bleparse.test.d.ts +0 -2
  413. package/dist/test/matter/platform-matter.bleparse.test.d.ts.map +0 -1
  414. package/dist/test/matter/platform-matter.bleparse.test.js +0 -43
  415. package/dist/test/matter/platform-matter.bleparse.test.js.map +0 -1
  416. package/dist/test/matter/platform-matter.cleanup.test.d.ts +0 -2
  417. package/dist/test/matter/platform-matter.cleanup.test.d.ts.map +0 -1
  418. package/dist/test/matter/platform-matter.cleanup.test.js +0 -70
  419. package/dist/test/matter/platform-matter.cleanup.test.js.map +0 -1
  420. package/dist/test/matter/platform-matter.keepstale.test.d.ts +0 -2
  421. package/dist/test/matter/platform-matter.keepstale.test.d.ts.map +0 -1
  422. package/dist/test/matter/platform-matter.keepstale.test.js +0 -27
  423. package/dist/test/matter/platform-matter.keepstale.test.js.map +0 -1
  424. package/dist/test/matter/platform-matter.logging.test.d.ts +0 -2
  425. package/dist/test/matter/platform-matter.logging.test.d.ts.map +0 -1
  426. package/dist/test/matter/platform-matter.logging.test.js +0 -29
  427. package/dist/test/matter/platform-matter.logging.test.js.map +0 -1
  428. package/dist/test/matter/platform-matter.mapping.test.d.ts +0 -2
  429. package/dist/test/matter/platform-matter.mapping.test.d.ts.map +0 -1
  430. package/dist/test/matter/platform-matter.mapping.test.js +0 -43
  431. package/dist/test/matter/platform-matter.mapping.test.js.map +0 -1
  432. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts +0 -2
  433. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts.map +0 -1
  434. package/dist/test/matter/platform-matter.openapi-mapping.test.js +0 -84
  435. package/dist/test/matter/platform-matter.openapi-mapping.test.js.map +0 -1
  436. package/dist/test/matter/platform-matter.test.d.ts +0 -2
  437. package/dist/test/matter/platform-matter.test.d.ts.map +0 -1
  438. package/dist/test/matter/platform-matter.test.js +0 -117
  439. package/dist/test/matter/platform-matter.test.js.map +0 -1
  440. package/dist/test/matter/platform-matter.unregister.test.d.ts +0 -2
  441. package/dist/test/matter/platform-matter.unregister.test.d.ts.map +0 -1
  442. package/dist/test/matter/platform-matter.unregister.test.js +0 -30
  443. package/dist/test/matter/platform-matter.unregister.test.js.map +0 -1
  444. package/dist/test/matter/platform-matter.webhook.test.d.ts +0 -2
  445. package/dist/test/matter/platform-matter.webhook.test.d.ts.map +0 -1
  446. package/dist/test/matter/platform-matter.webhook.test.js +0 -46
  447. package/dist/test/matter/platform-matter.webhook.test.js.map +0 -1
  448. package/dist/test/utils.test.d.ts +0 -2
  449. package/dist/test/utils.test.d.ts.map +0 -1
  450. package/dist/test/utils.test.js +0 -95
  451. package/dist/test/utils.test.js.map +0 -1
  452. package/dist/test/verifyconfig.test.d.ts +0 -2
  453. package/dist/test/verifyconfig.test.d.ts.map +0 -1
  454. package/dist/test/verifyconfig.test.js +0 -167
  455. package/dist/test/verifyconfig.test.js.map +0 -1
  456. package/src/custom.d.ts +0 -7
  457. package/src/devices-hap/airpurifier.ts +0 -568
  458. package/src/devices-hap/blindtilt.ts +0 -1049
  459. package/src/devices-hap/bot.ts +0 -910
  460. package/src/devices-hap/ceilinglight.ts +0 -747
  461. package/src/devices-hap/colorbulb.ts +0 -940
  462. package/src/devices-hap/contact.ts +0 -457
  463. package/src/devices-hap/curtain.ts +0 -944
  464. package/src/devices-hap/device.ts +0 -890
  465. package/src/devices-hap/fan.ts +0 -716
  466. package/src/devices-hap/hub.ts +0 -440
  467. package/src/devices-hap/humidifier.ts +0 -762
  468. package/src/devices-hap/iosensor.ts +0 -442
  469. package/src/devices-hap/lightstrip.ts +0 -863
  470. package/src/devices-hap/lock.ts +0 -627
  471. package/src/devices-hap/meter.ts +0 -427
  472. package/src/devices-hap/meterplus.ts +0 -431
  473. package/src/devices-hap/meterpro.ts +0 -523
  474. package/src/devices-hap/motion.ts +0 -390
  475. package/src/devices-hap/plug.ts +0 -427
  476. package/src/devices-hap/relayswitch.ts +0 -727
  477. package/src/devices-hap/robotvacuumcleaner.ts +0 -574
  478. package/src/devices-hap/waterdetector.ts +0 -400
  479. package/src/devices-matter/BaseMatterAccessory.ts +0 -302
  480. package/src/devices-matter/ColorLightAccessory.ts +0 -110
  481. package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -90
  482. package/src/devices-matter/ContactSensorAccessory.ts +0 -41
  483. package/src/devices-matter/DimmableLightAccessory.ts +0 -192
  484. package/src/devices-matter/DoorLockAccessory.ts +0 -60
  485. package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -122
  486. package/src/devices-matter/FanAccessory.ts +0 -95
  487. package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
  488. package/src/devices-matter/LeakSensorAccessory.ts +0 -40
  489. package/src/devices-matter/LightSensorAccessory.ts +0 -41
  490. package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
  491. package/src/devices-matter/OnOffLightAccessory.ts +0 -125
  492. package/src/devices-matter/OnOffOutletAccessory.ts +0 -51
  493. package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
  494. package/src/devices-matter/RoboticVacuumAccessory.ts +0 -621
  495. package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
  496. package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
  497. package/src/devices-matter/ThermostatAccessory.ts +0 -110
  498. package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
  499. package/src/devices-matter/WindowBlindAccessory.ts +0 -135
  500. package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
  501. package/src/devices-matter/custom/index.ts +0 -8
  502. package/src/devices-matter/index.ts +0 -29
  503. package/src/irdevice/airconditioner.ts +0 -533
  504. package/src/irdevice/airpurifier.ts +0 -252
  505. package/src/irdevice/camera.ts +0 -129
  506. package/src/irdevice/fan.ts +0 -226
  507. package/src/irdevice/irdevice.ts +0 -383
  508. package/src/irdevice/light.ts +0 -246
  509. package/src/irdevice/other.ts +0 -790
  510. package/src/irdevice/tv.ts +0 -378
  511. package/src/irdevice/vacuumcleaner.ts +0 -126
  512. package/src/irdevice/waterheater.ts +0 -127
  513. package/src/platform-hap.ts +0 -3193
  514. package/src/platform-matter.ts +0 -2703
  515. package/src/test/apiRequestTracker.test.ts +0 -417
  516. package/src/test/hap/device-webhook-context.test.ts +0 -136
  517. package/src/test/hap/platform-hap.logging.test.ts +0 -36
  518. package/src/test/hap/platform-hap.test.ts +0 -70
  519. package/src/test/helpers/platform-fixtures.ts +0 -33
  520. package/src/test/homebridge-ui/server.test.ts +0 -486
  521. package/src/test/index.test.ts +0 -24
  522. package/src/test/matter/devices-matter/baseMatterAccessory.test.ts +0 -88
  523. package/src/test/matter/platform-matter.additional.test.ts +0 -44
  524. package/src/test/matter/platform-matter.bleparse.test.ts +0 -47
  525. package/src/test/matter/platform-matter.cleanup.test.ts +0 -86
  526. package/src/test/matter/platform-matter.keepstale.test.ts +0 -37
  527. package/src/test/matter/platform-matter.logging.test.ts +0 -33
  528. package/src/test/matter/platform-matter.mapping.test.ts +0 -57
  529. package/src/test/matter/platform-matter.openapi-mapping.test.ts +0 -109
  530. package/src/test/matter/platform-matter.test.ts +0 -144
  531. package/src/test/matter/platform-matter.unregister.test.ts +0 -39
  532. package/src/test/matter/platform-matter.webhook.test.ts +0 -54
  533. package/src/test/utils.test.ts +0 -96
  534. package/src/test/verifyconfig.test.ts +0 -198
@@ -0,0 +1,73 @@
1
+ import { describe, it, expect } from 'vitest'
2
+ import { SwitchBotHAPPlatform, SwitchBotMatterPlatform } from '../src/platform'
3
+
4
+ const log = { info: () => {}, warn: () => {}, error: () => {}, debug: () => {} }
5
+
6
+ describe('accessory restore', () => {
7
+ it('reuses restored HAP accessory by context.deviceId and avoids re-register', async () => {
8
+ const registered: any[] = []
9
+ const hap = { uuid: { generate: (s: string) => `uuid-${s}` }, Service: {}, Characteristic: {} }
10
+
11
+ function PlatformAccessory(name: string, uuid: string) {
12
+ this.displayName = name
13
+ this.UUID = uuid
14
+ this.context = {}
15
+ this.getService = () => undefined
16
+ this.addService = () => undefined
17
+ }
18
+
19
+ const api: any = {
20
+ hap,
21
+ platformAccessory: PlatformAccessory as any,
22
+ registerPlatformAccessories: (_p: string, _n: string, accs: any[]) => registered.push(...accs),
23
+ on: (_ev: string, cb: Function) => {},
24
+ }
25
+
26
+ const platform = new SwitchBotHAPPlatform(log as any, { devices: [{ id: 'dev1', type: 'light', name: 'NewName' }] }, api)
27
+
28
+ // Simulate restored accessory with different uuid but correct deviceId in context
29
+ const restored = new (api.platformAccessory)('OldName', 'old-uuid')
30
+ restored.context = { deviceId: 'dev1', type: 'light' }
31
+ platform.configureAccessory(restored)
32
+
33
+ // Load devices should reuse restored accessory by context.deviceId
34
+ await (platform as any).loadDevices()
35
+
36
+ // No new registration for this device because restored accessory was reused
37
+ expect(registered.length).toBe(0)
38
+ // The restored accessory should remain in the platform's accessory map
39
+ const keys = Array.from((platform as any).accessories.keys())
40
+ expect(keys).toContain('old-uuid')
41
+ const acc = (platform as any).accessories.get('old-uuid')
42
+ expect(acc.context.deviceId).toBe('dev1')
43
+ })
44
+
45
+ it('reuses restored Matter accessory by context.deviceId when registering', async () => {
46
+ const captured: any[] = []
47
+ const matter = { uuid: { generate: (s: string) => `uuid-${s}` }, clusterNames: {} }
48
+ const api: any = {
49
+ matter,
50
+ isMatterAvailable: () => true,
51
+ isMatterEnabled: () => true,
52
+ on: (_ev: string, cb: Function) => {},
53
+ }
54
+
55
+ const platform = new SwitchBotMatterPlatform(log as any, { devices: [{ id: 'm1', type: 'light', name: 'M1' }] }, api)
56
+
57
+ // Simulate restored Matter accessory with a different uuid but matching deviceId
58
+ const restored = { uuid: 'restored-uuid', displayName: 'OldM', context: { deviceId: 'm1', type: 'light' } }
59
+ ;(platform as any).accessories.set(restored.uuid, restored)
60
+
61
+ // stub matter.registerPlatformAccessories to capture what is registered
62
+ api.matter.registerPlatformAccessories = async (_p: string, _n: string, accs: any[]) => captured.push(...accs)
63
+
64
+ // call registerMatterAccessories
65
+ await (platform as any).registerMatterAccessories()
66
+
67
+ // The restored accessory should still be present and associated with deviceId
68
+ const keys = Array.from((platform as any).accessories.keys())
69
+ expect(keys).toContain(restored.uuid)
70
+ const acc = (platform as any).accessories.get(restored.uuid)
71
+ expect(acc.context.deviceId).toBe('m1')
72
+ })
73
+ })
@@ -0,0 +1,37 @@
1
+ import { describe, it, expect } from 'vitest'
2
+ import { createDevice } from '../src/deviceFactory'
3
+
4
+ const DEVICE_TYPES = [
5
+ 'bot',
6
+ 'curtain',
7
+ 'fan',
8
+ 'light',
9
+ 'lightstrip',
10
+ 'motion',
11
+ 'contact',
12
+ 'vacuum',
13
+ 'lock',
14
+ 'humidifier',
15
+ 'temperature',
16
+ ]
17
+
18
+ describe('device mapping', () => {
19
+ it('maps known device types to device descriptors (HAP)', async () => {
20
+ for (const t of DEVICE_TYPES) {
21
+ const created = await createDevice({ id: `dev-${t}`, type: t, name: `Name-${t}` }, {}, false)
22
+ expect(created).toHaveProperty('instance')
23
+ expect(created).toHaveProperty('createAccessory')
24
+ const acc = created.createAccessory?.(undefined)
25
+ expect(acc).toBeDefined()
26
+ // descriptor should include services for HAP devices
27
+ if (created.protocol === 'hap') {
28
+ expect(acc).toHaveProperty('services')
29
+ }
30
+ }
31
+ })
32
+
33
+ it('returns matter protocol when requested', async () => {
34
+ const created = await createDevice({ id: 'm1', type: 'light', name: 'M1' }, { preferMatter: true, enableMatter: true }, true)
35
+ expect(created.protocol).toBe('matter')
36
+ })
37
+ })
@@ -0,0 +1,18 @@
1
+ import { describe, it, expect } from 'vitest'
2
+ import { createDevice } from '../src/deviceFactory'
3
+
4
+ describe('deviceFactory', () => {
5
+ it('creates HAP device descriptor for light', async () => {
6
+ const created = await createDevice({ id: 'dev1', type: 'light', name: 'L1' }, {}, false)
7
+ expect(created).toHaveProperty('instance')
8
+ expect(created).toHaveProperty('createAccessory')
9
+ expect(created.protocol).toBe('hap')
10
+ const acc = created.createAccessory?.(undefined)
11
+ expect(acc).toHaveProperty('services')
12
+ })
13
+
14
+ it('creates Matter descriptor when useMatter=true', async () => {
15
+ const created = await createDevice({ id: 'dev2', type: 'light', name: 'L2' }, { preferMatter: true, enableMatter: true }, true)
16
+ expect(created.protocol).toBe('matter')
17
+ })
18
+ })
@@ -0,0 +1,50 @@
1
+ import { describe, test } from 'vitest'
2
+ import { promisify } from 'util'
3
+ import { execFile } from 'child_process'
4
+ import path from 'path'
5
+
6
+ const execFileP = promisify(execFile)
7
+
8
+ describe('E2E scripts (manual-run harness)', () => {
9
+ if (!process.env.RUN_E2E) {
10
+ test.skip('E2E disabled - set RUN_E2E=true to enable', () => {})
11
+ return
12
+ }
13
+
14
+ test('run configured E2E scripts sequentially', async () => {
15
+ const repoRoot = path.resolve(__dirname, '../../')
16
+ const scriptsDir = path.join(repoRoot, 'scripts', 'e2e')
17
+
18
+ const scriptsToRun: { envVar: string; script: string }[] = [
19
+ { envVar: 'LIGHT_ACCESSORY_ID', script: 'light-e2e.sh' },
20
+ { envVar: 'FAN_ACCESSORY_ID', script: 'fan-e2e.sh' },
21
+ { envVar: 'CURTAIN_ACCESSORY_ID', script: 'curtain-e2e.sh' },
22
+ { envVar: 'LOCK_ACCESSORY_ID', script: 'lock-e2e.sh' },
23
+ ]
24
+
25
+ for (const s of scriptsToRun) {
26
+ if (!process.env[s.envVar]) {
27
+ // skip script if no accessory id configured
28
+ // eslint-disable-next-line no-console
29
+ console.log(`Skipping ${s.script} because ${s.envVar} not set`)
30
+ continue
31
+ }
32
+
33
+ const scriptPath = path.join(scriptsDir, s.script)
34
+ // Forward environment so scripts can read HB_URL / HB_TOKEN / <ACCESSORY_ID>
35
+ const env = Object.assign({}, process.env)
36
+
37
+ try {
38
+ // eslint-disable-next-line no-console
39
+ console.log(`Running ${s.script}...`)
40
+ const { stdout, stderr } = await execFileP('bash', [scriptPath], { env })
41
+ // eslint-disable-next-line no-console
42
+ console.log(stdout)
43
+ if (stderr) console.error(stderr)
44
+ } catch (e: any) {
45
+ // bubble up as test failure
46
+ throw new Error(`Script ${s.script} failed: ${e.message}`)
47
+ }
48
+ }
49
+ })
50
+ })
@@ -0,0 +1,29 @@
1
+ import { describe, it, expect, vi } from 'vitest'
2
+ import { FanDevice } from '../src/devices/genericDevice'
3
+
4
+ describe('Fan swing commands', () => {
5
+ it('sends setSwing on boolean swing', async () => {
6
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
7
+ const d = new FanDevice({ id: 'fan1', type: 'fan', name: 'F' }, { _client: mockClient } as any)
8
+ await d.init()
9
+ const res = await d.setState({ swing: true })
10
+ expect(mockClient.setDeviceState).toHaveBeenCalled()
11
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'setSwing', parameter: 'on', commandType: 'command' })
12
+ })
13
+
14
+ it('sends setSwing with angle', async () => {
15
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
16
+ const d = new FanDevice({ id: 'fan2', type: 'fan', name: 'F2' }, { _client: mockClient } as any)
17
+ await d.init()
18
+ await d.setState({ swingAngle: 45 })
19
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'setSwing', parameter: '45', commandType: 'command' })
20
+ })
21
+
22
+ it('sends setSwing with mode', async () => {
23
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
24
+ const d = new FanDevice({ id: 'fan3', type: 'fan', name: 'F3' }, { _client: mockClient } as any)
25
+ await d.init()
26
+ await d.setState({ swingMode: 'vertical' })
27
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'setSwing', parameter: 'vertical', commandType: 'command' })
28
+ })
29
+ })
@@ -0,0 +1,53 @@
1
+ export function makeFakeMatterApi() {
2
+ const registered: any[] = []
3
+ return {
4
+ uuid: { generate: (s: string) => `m-${s}` },
5
+ registerPlatformAccessories: async (_plugin: string, _name: string, accs: any[]) => {
6
+ registered.push(...accs)
7
+ },
8
+ getRegistered: () => registered,
9
+ }
10
+ }
11
+
12
+ export function makeFakeHap() {
13
+ return {
14
+ uuid: { generate: (s: string) => `h-${s}` },
15
+ Service: { Switch: 'Switch', Lightbulb: 'Lightbulb', Fan: 'Fan', WindowCovering: 'WindowCovering', LockMechanism: 'LockMechanism', HumiditySensor: 'HumiditySensor' },
16
+ Characteristic: { On: 'On', Brightness: 'Brightness', RotationSpeed: 'RotationSpeed', CurrentPosition: 'CurrentPosition', TargetPosition: 'TargetPosition' },
17
+ }
18
+ }
19
+
20
+ export function makeFakeApiWithMatter(matterApi?: any) {
21
+ const matter = matterApi || makeFakeMatterApi()
22
+ const api: any = {
23
+ matter,
24
+ isMatterAvailable: () => true,
25
+ isMatterEnabled: () => true,
26
+ hap: makeFakeHap(),
27
+ }
28
+ return api
29
+ }
30
+
31
+ export function makeFakeApiWithoutMatter() {
32
+ // Provide a minimal platformAccessory constructor and registerPlatformAccessories
33
+ function PlatformAccessory(name: string, uuid: string) {
34
+ this.displayName = name
35
+ this.UUID = uuid
36
+ this.services = []
37
+ this.getService = (type: any) => this.services.find((s: any) => s.type === type)
38
+ this.addService = (type: any) => {
39
+ const s: any = { type, characteristics: {} }
40
+ this.services.push(s)
41
+ return s
42
+ }
43
+ }
44
+
45
+ return {
46
+ isMatterAvailable: () => false,
47
+ isMatterEnabled: () => false,
48
+ hap: makeFakeHap(),
49
+ platformAccessory: PlatformAccessory,
50
+ registerPlatformAccessories: (_p: string, _n: string, accs: any[]) => {},
51
+ on: (_ev: string, cb: Function) => {},
52
+ }
53
+ }
@@ -0,0 +1,44 @@
1
+ import { describe, it, expect, vi } from 'vitest'
2
+ import { LockDevice } from '../src/devices/genericDevice'
3
+
4
+ describe('Lock user management', () => {
5
+ it('sets lock pin via pin field', async () => {
6
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
7
+ const d = new LockDevice({ id: 'lock1', type: 'lock', name: 'L1' }, { _client: mockClient } as any)
8
+ await d.init()
9
+ await d.setState({ pin: '1234' })
10
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'setLockPin', parameter: '1234', commandType: 'command' })
11
+ })
12
+
13
+ it('adds a user code with addUser action', async () => {
14
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
15
+ const d = new LockDevice({ id: 'lock2', type: 'lock', name: 'L2' }, { _client: mockClient } as any)
16
+ await d.init()
17
+ await d.setState({ action: 'addUser', user: 'user1', pin: '9999' })
18
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'addUserCode', parameter: 'user1:9999', commandType: 'command' })
19
+ })
20
+
21
+ it('removes a user code with removeUser action', async () => {
22
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
23
+ const d = new LockDevice({ id: 'lock3', type: 'lock', name: 'L3' }, { _client: mockClient } as any)
24
+ await d.init()
25
+ await d.setState({ action: 'removeUser', user: 'user1' })
26
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'removeUserCode', parameter: 'user1', commandType: 'command' })
27
+ })
28
+
29
+ it('lists users with listUsers action', async () => {
30
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
31
+ const d = new LockDevice({ id: 'lock4', type: 'lock', name: 'L4' }, { _client: mockClient } as any)
32
+ await d.init()
33
+ await d.setState({ action: 'listUsers' })
34
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'listUsers', parameter: 'default', commandType: 'command' })
35
+ })
36
+
37
+ it('unlocks with pin using unlockWithPin action', async () => {
38
+ const mockClient = { setDeviceState: vi.fn(async (id: string, body: any) => ({ id, body })) }
39
+ const d = new LockDevice({ id: 'lock5', type: 'lock', name: 'L5' }, { _client: mockClient } as any)
40
+ await d.init()
41
+ await d.setState({ action: 'unlockWithPin', pin: '5555' })
42
+ expect((mockClient.setDeviceState as any).mock.calls[0][1]).toEqual({ command: 'unlockWithPin', parameter: '5555', commandType: 'command' })
43
+ })
44
+ })
@@ -0,0 +1,55 @@
1
+ import { describe, it, expect, beforeEach } from 'vitest'
2
+ import SwitchBotPlatform from '../src/platform'
3
+ import { createPlatformProxy } from '../src/utils'
4
+
5
+ // This is a lightweight integration-style unit test that verifies the
6
+ // platform will attempt Matter registration when MatterPlatform is
7
+ // provided by the proxy and will still restore cached accessories.
8
+
9
+ describe('platform integration (Matter)', () => {
10
+ it('creates Matter descriptors when useMatter=true and restores accessories', async () => {
11
+ // Create a fake MatterPlatform class that records register calls
12
+ class FakeMatterPlatform {
13
+ public registered: any[] = []
14
+ constructor(public log: any, public cfg: any, public api: any) {}
15
+ async registerMatterAccessories(accessories: any[]) {
16
+ this.registered.push(...accessories)
17
+ return
18
+ }
19
+ // simulate configureAccessory hook for restored matter accessories
20
+ configureMatterAccessory(accessory: any) {
21
+ // no-op
22
+ }
23
+ // lightweight loadDevices implementation that uses an injected discover hook if present
24
+ async loadDevices() {
25
+ const devices = (this.cfg?.devices) || (this as any)._discoverDevices ? await (this as any)._discoverDevices() : []
26
+ // Simulate preparing descriptors and immediately registering them
27
+ const serialized = devices.map((d: any) => ({ id: d.id ?? d.deviceId ?? d, protocol: 'matter', name: d.name }))
28
+ if (serialized.length > 0) await this.registerMatterAccessories(serialized)
29
+ }
30
+ }
31
+
32
+ // Create proxy that would choose the fake matter platform when preferred
33
+ const Proxy = createPlatformProxy(function HAP() {}, FakeMatterPlatform)
34
+ const platform = new (Proxy as any)(console, { preferMatter: true, enableMatter: true }, {})
35
+
36
+ // Make a small device descriptor to load
37
+ const device = { id: 'dev-m-1', type: 'light', name: 'MatterLight' }
38
+ // Simulate platform method that loads devices (use same method name used in code)
39
+ if (typeof platform.loadDevices === 'function') {
40
+ // Add an internal method to return our single device for creation
41
+ platform._discoverDevices = async () => [device]
42
+ // Call loadDevices which should call the MatterPlatform.registerMatterAccessories
43
+ await platform.loadDevices()
44
+
45
+ // Verify FakeMatterPlatform received the accessory
46
+ expect(platform).toBeInstanceOf(FakeMatterPlatform)
47
+ expect((platform as any).registered.length).toBeGreaterThanOrEqual(1)
48
+ const registered = (platform as any).registered[0]
49
+ expect(registered).toHaveProperty('protocol', 'matter')
50
+ expect(registered).toHaveProperty('id', device.id)
51
+ } else {
52
+ throw new Error('platform.loadDevices not implemented')
53
+ }
54
+ })
55
+ })
@@ -0,0 +1,97 @@
1
+ import { describe, it, expect, vi, beforeEach } from 'vitest'
2
+ import { FanDevice, LightDevice, CurtainDevice, LockDevice } from '../src/devices/genericDevice'
3
+
4
+ describe('Matter descriptors and translations (per-device)', () => {
5
+ beforeEach(() => {
6
+ vi.restoreAllMocks()
7
+ })
8
+
9
+ it('Fan: maps oscillation and swing to setDeviceState commands', async () => {
10
+ const calls: any[] = []
11
+ const fakeClient: any = { setDeviceState: async (id: string, body: any) => { calls.push({ id, body }); return { ok: true } } }
12
+ const d = new FanDevice({ id: 'f1', type: 'fan' } as any, { _client: fakeClient } as any)
13
+
14
+ // Oscillate
15
+ await d.setState({ oscillate: true })
16
+ expect(calls.pop()?.body.command).toBe('setOscillation')
17
+
18
+ // Swing boolean
19
+ await d.setState({ swing: true })
20
+ expect(calls.pop()?.body.command).toBe('setSwing')
21
+
22
+ // Swing angle
23
+ await d.setState({ swingAngle: 45 })
24
+ const last = calls.pop()
25
+ expect(last.body.command).toBe('setSwing')
26
+ expect(last.body.parameter).toBe('45')
27
+ })
28
+
29
+ it('Light: color/CT mapping and setState commands', async () => {
30
+ const calls: any[] = []
31
+ const fakeClient: any = { setDeviceState: async (id: string, body: any) => { calls.push(body); return { ok: true } } }
32
+ const d = new LightDevice({ id: 'l1', type: 'light' } as any, { _client: fakeClient } as any)
33
+
34
+ // color temperature (mired)
35
+ await d.setState({ colorTemperature: 300 })
36
+ expect(calls.pop()?.command).toBe('setColorTemperature')
37
+
38
+ // hue + saturation
39
+ await d.setState({ hue: 120, saturation: 50 })
40
+ const c = calls.pop()
41
+ expect(c.command).toBe('setColor')
42
+ expect(c.parameter).toBe('120,50')
43
+
44
+ // hex color string
45
+ await d.setState({ color: '#ff0000' })
46
+ expect(calls.pop()?.command).toBe('setColor')
47
+ })
48
+
49
+ it('Curtain: position mapping and Matter attribute writes', async () => {
50
+ const calls: any[] = []
51
+ const fakeClient: any = { setDeviceState: async (id: string, body: any) => { calls.push(body); return { ok: true } } }
52
+ const d = new CurtainDevice({ id: 'c1', type: 'curtain' } as any, { _client: fakeClient } as any)
53
+
54
+ // Setting position via HAP-style setState
55
+ await d.setState({ position: 25 })
56
+ expect(calls.pop()?.command).toBe('setPosition')
57
+
58
+ // Matter descriptor write: call targetPosition write handler
59
+ const matter = d.createMatterAccessory({})
60
+ const cluster = matter.clusters.find((c: any) => c.type === 'Shade')
61
+ // targetPosition attribute exists and has a write function
62
+ expect(cluster.attributes.targetPosition.write).toBeTypeOf('function')
63
+ await cluster.attributes.targetPosition.write(80)
64
+ expect(calls.pop()?.command).toBe('setPosition')
65
+ })
66
+
67
+ it('Lock: user management actions map to correct commands', async () => {
68
+ const calls: any[] = []
69
+ const fakeClient: any = { setDeviceState: async (id: string, body: any) => { calls.push(body); return { ok: true } } }
70
+ const d = new LockDevice({ id: 'lock1', type: 'lock' } as any, { _client: fakeClient } as any)
71
+
72
+ // addUser via setState
73
+ await d.setState({ action: 'addUser', user: 'u1', pin: '1234' })
74
+ let last = calls.pop()
75
+ expect(last.command).toBe('addUserCode')
76
+ expect(last.parameter).toBe('u1:1234')
77
+
78
+ // removeUser
79
+ await d.setState({ action: 'removeUser', user: 'u1' })
80
+ last = calls.pop()
81
+ expect(last.command).toBe('removeUserCode')
82
+
83
+ // unlockWithPin
84
+ await d.setState({ action: 'unlockWithPin', pin: '0000' })
85
+ last = calls.pop()
86
+ expect(last.command).toBe('unlockWithPin')
87
+
88
+ // Matter descriptor write for addUser
89
+ const matter = d.createMatterAccessory({})
90
+ const mgmt = matter.clusters.find((c: any) => c.type === 'DoorLockUserManagement')
91
+ expect(mgmt.attributes.addUser.write).toBeTypeOf('function')
92
+ await mgmt.attributes.addUser.write({ user: 'u2', pin: '4321' })
93
+ last = calls.pop()
94
+ expect(last.command).toBe('addUserCode')
95
+ expect(last.parameter).toBe('u2:4321')
96
+ })
97
+ })
@@ -0,0 +1,101 @@
1
+ import { describe, it, expect } from 'vitest'
2
+ import { LightDevice, FanDevice, CurtainDevice, LockDevice, HumidifierMatterDevice } from '../src/devices/genericDevice'
3
+ import { makeFakeApiWithMatter } from './helpers/matter-harness'
4
+
5
+ const api = makeFakeApiWithMatter()
6
+
7
+ describe('Matter device state translations', () => {
8
+ it('Light: reads and writes map to setState/getState', async () => {
9
+ const device = new LightDevice({ id: 'l-1', type: 'light', name: 'L1' }, {})
10
+ let lastSet: any = null
11
+ device.getState = async () => ({ on: true, brightness: 75, hue: 120, saturation: 50, colorTemperature: 350 })
12
+ device.setState = async (c: any) => { lastSet = c; return { success: true } }
13
+
14
+ const desc = device.createMatterAccessory(api)
15
+ const levelCluster = desc.clusters.find((c: any) => c.clusterId === 0x0008 || c.type === 'LevelControl')
16
+ expect(levelCluster).toBeDefined()
17
+ const levelAttr = levelCluster.attributes.currentLevel || levelCluster.attributes["0x0000"] || Object.values(levelCluster.attributes)[0]
18
+ const read = await levelAttr.read()
19
+ expect(read).toBe(75)
20
+
21
+ // write should call setState with brightness
22
+ await levelAttr.write(30)
23
+ expect(lastSet).toMatchObject({ brightness: 30 })
24
+ })
25
+
26
+ it('Fan: reads speed and writes speed/oscillation', async () => {
27
+ const device = new FanDevice({ id: 'f-1', type: 'fan', name: 'F1' }, {})
28
+ let last: any = null
29
+ device.getState = async () => ({ on: false, speed: 45, oscillating: true, swingMode: 'auto' })
30
+ device.setState = async (c: any) => { last = c; return { success: true } }
31
+
32
+ const desc = device.createMatterAccessory(api)
33
+ const fanCluster = desc.clusters.find((c: any) => c.clusterId === 0x0202 || c.type === 'FanControl')
34
+ expect(fanCluster).toBeDefined()
35
+ const speedAttr = fanCluster.attributes.rotationSpeed || Object.values(fanCluster.attributes)[0]
36
+ const speed = await speedAttr.read()
37
+ expect(speed).toBe(45)
38
+
39
+ await speedAttr.write(80)
40
+ expect(last).toMatchObject({ speed: 80 })
41
+
42
+ // oscillation write
43
+ const osc = fanCluster.attributes.oscillation
44
+ await osc.write(true)
45
+ expect(last).toMatchObject({ oscillate: true })
46
+ })
47
+
48
+ it('Curtain: reads and writes position', async () => {
49
+ const device = new CurtainDevice({ id: 'c-1', type: 'curtain', name: 'C1' }, {})
50
+ let last: any = null
51
+ device.getState = async () => ({ position: 42 })
52
+ device.setState = async (c: any) => { last = c; return { success: true } }
53
+
54
+ const desc = device.createMatterAccessory(api)
55
+ const shade = desc.clusters.find((c: any) => c.clusterId === 0x0102 || c.type === 'Shade')
56
+ expect(shade).toBeDefined()
57
+ const current = shade.attributes.currentPosition || shade.attributes["0x0000"]
58
+ const v = await current.read()
59
+ expect(v).toBe(42)
60
+
61
+ const target = shade.attributes.targetPosition || shade.attributes["0x0001"]
62
+ await target.write(10)
63
+ expect(last).toMatchObject({ position: 10 })
64
+ })
65
+
66
+ it('Lock: reads and writes locked state', async () => {
67
+ const device = new LockDevice({ id: 'k-1', type: 'lock', name: 'K1' }, {})
68
+ let last: any = null
69
+ device.getState = async () => ({ locked: true })
70
+ device.setState = async (c: any) => { last = c; return { success: true } }
71
+
72
+ const desc = device.createMatterAccessory(api)
73
+ const lock = desc.clusters.find((c: any) => c.clusterId === 0x0101 || c.type === 'DoorLock')
74
+ expect(lock).toBeDefined()
75
+ const lockAttr = lock.attributes.lockState || lock.attributes["0x0000"]
76
+ const v = await lockAttr.read()
77
+ expect(v).toBe(true)
78
+
79
+ await lockAttr.write(false)
80
+ expect(last).toMatchObject({ locked: false })
81
+ })
82
+
83
+ it('Humidifier: reads humidity and toggles on/off', async () => {
84
+ const device = new HumidifierMatterDevice({ id: 'h-1', type: 'humidifier', name: 'H1' }, {})
85
+ let last: any = null
86
+ device.getState = async () => ({ humidity: 55, on: false })
87
+ device.setState = async (c: any) => { last = c; return { success: true } }
88
+
89
+ const desc = device.createMatterAccessory(api)
90
+ const rh = desc.clusters.find((c: any) => c.clusterId === 0x0405 || c.type === 'RelativeHumiditySensor')
91
+ expect(rh).toBeDefined()
92
+ const humAttr = rh.attributes.currentRelativeHumidity || rh.attributes["0x0000"]
93
+ const hv = await humAttr.read()
94
+ expect(hv).toBe(55)
95
+
96
+ const onCluster = desc.clusters.find((c: any) => c.clusterId === 0x0006 || c.type === 'OnOff')
97
+ const onAttr = onCluster.attributes.onOff || onCluster.attributes["0x0000"]
98
+ await onAttr.write(true)
99
+ expect(last).toMatchObject({ on: true })
100
+ })
101
+ })
@@ -0,0 +1,70 @@
1
+ import { describe, it, expect } from 'vitest'
2
+ import { createPlatformProxy } from '../src/utils'
3
+ import { makeFakeApiWithMatter, makeFakeApiWithoutMatter, makeFakeMatterApi } from './helpers/matter-harness'
4
+ import SwitchBotHAPPlatform, { SwitchBotMatterPlatform } from '../src/platform'
5
+
6
+ // Expanded Matter integration harness tests
7
+
8
+ describe('Matter integration harness', () => {
9
+ it('registers Matter accessories when Matter API available', async () => {
10
+ // Instantiate the real Matter platform directly to test its registration path
11
+ const matterApi = makeFakeMatterApi()
12
+ const api = makeFakeApiWithMatter(matterApi)
13
+
14
+ const platform = new SwitchBotMatterPlatform(console as any, { devices: [{ id: 'm-1', type: 'light', name: 'L1' }], enableMatter: true, preferMatter: true } as any, api as any)
15
+
16
+ // Call register path directly
17
+ if (typeof (platform as any).registerMatterAccessories === 'function') {
18
+ await (platform as any).registerMatterAccessories()
19
+ } else if (typeof (platform as any).loadDevices === 'function') {
20
+ await (platform as any).loadDevices()
21
+ }
22
+
23
+ const regs = matterApi.getRegistered()
24
+ expect(regs.length).toBeGreaterThanOrEqual(1)
25
+ // The platform registers serialized accessories; ensure deviceId is present in context
26
+ expect(regs[0]).toHaveProperty('context')
27
+ expect(regs[0].context).toHaveProperty('deviceId', 'm-1')
28
+ })
29
+
30
+ it('falls back to HAP when Matter API unavailable', async () => {
31
+ // Directly instantiate the HAP platform to exercise HAP fallback
32
+ const api = makeFakeApiWithoutMatter()
33
+ const platform = new SwitchBotHAPPlatform(console as any, { devices: [{ id: 'h-1', type: 'light', name: 'H1' }], enableMatter: true, preferMatter: true } as any, api as any)
34
+
35
+ // call loadDevices for HAP flow
36
+ if (typeof (platform as any).loadDevices === 'function') await (platform as any).loadDevices()
37
+
38
+ // Platform should have an accessories map populated
39
+ const accMap = (platform as any).accessories
40
+ expect(accMap).toBeDefined()
41
+ const keys = Array.from(accMap.keys())
42
+ expect(keys.length).toBeGreaterThanOrEqual(1)
43
+ })
44
+
45
+ it('reuses restored Matter accessories instead of registering duplicates', async () => {
46
+ const matterApi = makeFakeMatterApi()
47
+ const api = makeFakeApiWithMatter(matterApi)
48
+
49
+ const platform = new SwitchBotMatterPlatform(console as any, { devices: [{ id: 'r-1', type: 'light', name: 'R1' }], enableMatter: true, preferMatter: true } as any, api as any)
50
+
51
+ // Simulate a restored accessory present before registration
52
+ const restored = { uuid: matterApi.uuid.generate('r-1'), displayName: 'Restored R1', context: { deviceId: 'r-1' } }
53
+ if (typeof (platform as any).configureMatterAccessory === 'function') {
54
+ await (platform as any).configureMatterAccessory(restored)
55
+ } else if (typeof (platform as any).configureAccessory === 'function') {
56
+ await (platform as any).configureAccessory(restored)
57
+ }
58
+
59
+ // Now call register path
60
+ if (typeof (platform as any).registerMatterAccessories === 'function') await (platform as any).registerMatterAccessories()
61
+ else if (typeof (platform as any).loadDevices === 'function') {
62
+ await (platform as any).loadDevices()
63
+ if (typeof (platform as any).registerMatterAccessories === 'function') await (platform as any).registerMatterAccessories()
64
+ }
65
+
66
+ const accs = (platform as any).accessories
67
+ const found = Array.from(accs.values()).some((a: any) => a && (a.uuid === restored.uuid || a.UUID === restored.uuid))
68
+ expect(found).toBeTruthy()
69
+ })
70
+ })