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

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