@switchbot/homebridge-switchbot 5.0.0-beta.9 → 5.0.0-beta.90

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 (448) hide show
  1. package/.github/ISSUE_TEMPLATE/e2e-verification.md +36 -0
  2. package/.github/workflows/ci.yml +32 -0
  3. package/.github/workflows/manual-e2e.yml +115 -0
  4. package/.github/workflows/release.yml +0 -4
  5. package/CHANGELOG.md +35 -0
  6. package/E2E-VERIFICATION.md +121 -0
  7. package/MIGRATION.md +44 -0
  8. package/README.md +56 -3
  9. package/config.schema.json +91 -14787
  10. package/dist/deviceFactory.d.ts +13 -0
  11. package/dist/deviceFactory.d.ts.map +1 -0
  12. package/dist/deviceFactory.js +81 -0
  13. package/dist/deviceFactory.js.map +1 -0
  14. package/dist/devices/deviceBase.d.ts +50 -0
  15. package/dist/devices/deviceBase.d.ts.map +1 -0
  16. package/dist/devices/deviceBase.js +119 -0
  17. package/dist/devices/deviceBase.js.map +1 -0
  18. package/dist/devices/genericDevice.d.ts +283 -0
  19. package/dist/devices/genericDevice.d.ts.map +1 -0
  20. package/dist/devices/genericDevice.js +1035 -0
  21. package/dist/devices/genericDevice.js.map +1 -0
  22. package/dist/homebridge-ui/public/index.html +630 -246
  23. package/dist/homebridge-ui/server.d.ts +3 -1
  24. package/dist/homebridge-ui/server.d.ts.map +1 -1
  25. package/dist/homebridge-ui/server.js +367 -36
  26. package/dist/homebridge-ui/server.js.map +1 -1
  27. package/dist/index.d.ts.map +1 -1
  28. package/dist/index.js +4 -32
  29. package/dist/index.js.map +1 -1
  30. package/dist/platform.d.ts +35 -0
  31. package/dist/platform.d.ts.map +1 -0
  32. package/dist/platform.js +514 -0
  33. package/dist/platform.js.map +1 -0
  34. package/dist/settings.d.ts +10 -249
  35. package/dist/settings.d.ts.map +1 -1
  36. package/dist/settings.js +5 -30
  37. package/dist/settings.js.map +1 -1
  38. package/dist/switchbotClient.d.ts +32 -0
  39. package/dist/switchbotClient.d.ts.map +1 -0
  40. package/dist/switchbotClient.js +194 -0
  41. package/dist/switchbotClient.js.map +1 -0
  42. package/dist/utils.d.ts +39 -50
  43. package/dist/utils.d.ts.map +1 -1
  44. package/dist/utils.js +39 -688
  45. package/dist/utils.js.map +1 -1
  46. package/docs/assets/highlight.css +14 -0
  47. package/docs/assets/icons.js +1 -1
  48. package/docs/assets/icons.svg +1 -1
  49. package/docs/assets/main.js +2 -2
  50. package/docs/assets/style.css +3 -3
  51. package/docs/index.html +77 -13
  52. package/docs/variables/default.html +1 -1
  53. package/eslint.config.js +2 -8
  54. package/package.json +21 -28
  55. package/scripts/e2e/README.md +25 -0
  56. package/scripts/e2e/curtain-e2e.sh +70 -0
  57. package/scripts/e2e/fan-e2e.sh +75 -0
  58. package/scripts/e2e/light-advanced-e2e.sh +97 -0
  59. package/scripts/e2e/light-e2e.sh +75 -0
  60. package/scripts/e2e/list-accessories.sh +19 -0
  61. package/scripts/e2e/lock-e2e.sh +65 -0
  62. package/scripts/generate-matter-maps.js +60 -0
  63. package/scripts/run-e2e-local.sh +14 -0
  64. package/src/deviceFactory.ts +122 -0
  65. package/src/devices/deviceBase.ts +141 -0
  66. package/src/devices/genericDevice.ts +965 -0
  67. package/src/homebridge-ui/public/index.html +630 -246
  68. package/src/homebridge-ui/server.ts +434 -41
  69. package/src/index.ts +4 -33
  70. package/src/platform.ts +515 -0
  71. package/src/settings.ts +12 -277
  72. package/src/switchbotClient.ts +203 -0
  73. package/src/utils.ts +45 -713
  74. package/test/accessory-restore.spec.ts +73 -0
  75. package/test/device-mapping.spec.ts +37 -0
  76. package/test/deviceFactory.spec.ts +18 -0
  77. package/test/e2e/run-e2e.spec.ts +50 -0
  78. package/test/fan-swing.spec.ts +29 -0
  79. package/test/helpers/matter-harness.ts +53 -0
  80. package/test/lock-users.spec.ts +44 -0
  81. package/test/matter-childbridge.spec.ts +55 -0
  82. package/test/matter-descriptors.spec.ts +97 -0
  83. package/test/matter-device-state.spec.ts +101 -0
  84. package/test/matter-integration.spec.ts +70 -0
  85. package/test/platform.integration.spec.ts +55 -0
  86. package/test/switchbot-client-debounce.spec.ts +131 -0
  87. package/test/switchbot-client-openapi.spec.ts +56 -0
  88. package/test/switchbotClient.spec.ts +10 -0
  89. package/test/utils.spec.ts +20 -0
  90. package/vitest.config.ts +7 -0
  91. package/coverage/base.css +0 -224
  92. package/coverage/block-navigation.js +0 -87
  93. package/coverage/clover.xml +0 -15847
  94. package/coverage/coverage-final.json +0 -42
  95. package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
  96. package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
  97. package/coverage/docs/assets/dmt/index.html +0 -131
  98. package/coverage/docs/assets/hierarchy.js.html +0 -85
  99. package/coverage/docs/assets/icons.js.html +0 -136
  100. package/coverage/docs/assets/index.html +0 -146
  101. package/coverage/docs/assets/main.js.html +0 -265
  102. package/coverage/favicon.png +0 -0
  103. package/coverage/index.html +0 -191
  104. package/coverage/prettify.css +0 -1
  105. package/coverage/prettify.js +0 -2
  106. package/coverage/sort-arrow-sprite.png +0 -0
  107. package/coverage/sorter.js +0 -196
  108. package/coverage/src/device/blindtilt.ts.html +0 -3238
  109. package/coverage/src/device/bot.ts.html +0 -2803
  110. package/coverage/src/device/ceilinglight.ts.html +0 -2338
  111. package/coverage/src/device/colorbulb.ts.html +0 -2824
  112. package/coverage/src/device/contact.ts.html +0 -1465
  113. package/coverage/src/device/curtain.ts.html +0 -2869
  114. package/coverage/src/device/device.ts.html +0 -2500
  115. package/coverage/src/device/fan.ts.html +0 -2242
  116. package/coverage/src/device/hub.ts.html +0 -1408
  117. package/coverage/src/device/humidifier.ts.html +0 -2116
  118. package/coverage/src/device/index.html +0 -416
  119. package/coverage/src/device/iosensor.ts.html +0 -1375
  120. package/coverage/src/device/lightstrip.ts.html +0 -2617
  121. package/coverage/src/device/lock.ts.html +0 -1963
  122. package/coverage/src/device/meter.ts.html +0 -1372
  123. package/coverage/src/device/meterplus.ts.html +0 -1384
  124. package/coverage/src/device/meterpro.ts.html +0 -1618
  125. package/coverage/src/device/motion.ts.html +0 -1264
  126. package/coverage/src/device/plug.ts.html +0 -1372
  127. package/coverage/src/device/relayswitch.ts.html +0 -2284
  128. package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
  129. package/coverage/src/device/waterdetector.ts.html +0 -1294
  130. package/coverage/src/homebridge-ui/index.html +0 -116
  131. package/coverage/src/homebridge-ui/server.ts.html +0 -229
  132. package/coverage/src/index.html +0 -161
  133. package/coverage/src/index.ts.html +0 -124
  134. package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
  135. package/coverage/src/irdevice/airpurifier.ts.html +0 -844
  136. package/coverage/src/irdevice/camera.ts.html +0 -475
  137. package/coverage/src/irdevice/fan.ts.html +0 -766
  138. package/coverage/src/irdevice/index.html +0 -251
  139. package/coverage/src/irdevice/irdevice.ts.html +0 -1117
  140. package/coverage/src/irdevice/light.ts.html +0 -826
  141. package/coverage/src/irdevice/other.ts.html +0 -2458
  142. package/coverage/src/irdevice/tv.ts.html +0 -1222
  143. package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
  144. package/coverage/src/irdevice/waterheater.ts.html +0 -469
  145. package/coverage/src/platform.ts.html +0 -8776
  146. package/coverage/src/settings.ts.html +0 -934
  147. package/coverage/src/utils.ts.html +0 -2092
  148. package/dist/devices-hap/airpurifier.d.ts +0 -54
  149. package/dist/devices-hap/airpurifier.d.ts.map +0 -1
  150. package/dist/devices-hap/airpurifier.js +0 -527
  151. package/dist/devices-hap/airpurifier.js.map +0 -1
  152. package/dist/devices-hap/blindtilt.d.ts +0 -90
  153. package/dist/devices-hap/blindtilt.d.ts.map +0 -1
  154. package/dist/devices-hap/blindtilt.js +0 -974
  155. package/dist/devices-hap/blindtilt.js.map +0 -1
  156. package/dist/devices-hap/bot.d.ts +0 -102
  157. package/dist/devices-hap/bot.d.ts.map +0 -1
  158. package/dist/devices-hap/bot.js +0 -811
  159. package/dist/devices-hap/bot.js.map +0 -1
  160. package/dist/devices-hap/ceilinglight.d.ts +0 -85
  161. package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
  162. package/dist/devices-hap/ceilinglight.js +0 -701
  163. package/dist/devices-hap/ceilinglight.js.map +0 -1
  164. package/dist/devices-hap/colorbulb.d.ts +0 -88
  165. package/dist/devices-hap/colorbulb.d.ts.map +0 -1
  166. package/dist/devices-hap/colorbulb.js +0 -881
  167. package/dist/devices-hap/colorbulb.js.map +0 -1
  168. package/dist/devices-hap/contact.d.ts +0 -44
  169. package/dist/devices-hap/contact.d.ts.map +0 -1
  170. package/dist/devices-hap/contact.js +0 -409
  171. package/dist/devices-hap/contact.js.map +0 -1
  172. package/dist/devices-hap/curtain.d.ts +0 -73
  173. package/dist/devices-hap/curtain.d.ts.map +0 -1
  174. package/dist/devices-hap/curtain.js +0 -869
  175. package/dist/devices-hap/curtain.js.map +0 -1
  176. package/dist/devices-hap/device.d.ts +0 -98
  177. package/dist/devices-hap/device.d.ts.map +0 -1
  178. package/dist/devices-hap/device.js +0 -749
  179. package/dist/devices-hap/device.js.map +0 -1
  180. package/dist/devices-hap/fan.d.ts +0 -69
  181. package/dist/devices-hap/fan.d.ts.map +0 -1
  182. package/dist/devices-hap/fan.js +0 -649
  183. package/dist/devices-hap/fan.js.map +0 -1
  184. package/dist/devices-hap/hub.d.ts +0 -37
  185. package/dist/devices-hap/hub.d.ts.map +0 -1
  186. package/dist/devices-hap/hub.js +0 -392
  187. package/dist/devices-hap/hub.js.map +0 -1
  188. package/dist/devices-hap/humidifier.d.ts +0 -68
  189. package/dist/devices-hap/humidifier.d.ts.map +0 -1
  190. package/dist/devices-hap/humidifier.js +0 -628
  191. package/dist/devices-hap/humidifier.js.map +0 -1
  192. package/dist/devices-hap/iosensor.d.ts +0 -42
  193. package/dist/devices-hap/iosensor.d.ts.map +0 -1
  194. package/dist/devices-hap/iosensor.js +0 -382
  195. package/dist/devices-hap/iosensor.js.map +0 -1
  196. package/dist/devices-hap/lightstrip.d.ts +0 -79
  197. package/dist/devices-hap/lightstrip.d.ts.map +0 -1
  198. package/dist/devices-hap/lightstrip.js +0 -797
  199. package/dist/devices-hap/lightstrip.js.map +0 -1
  200. package/dist/devices-hap/lock.d.ts +0 -53
  201. package/dist/devices-hap/lock.d.ts.map +0 -1
  202. package/dist/devices-hap/lock.js +0 -561
  203. package/dist/devices-hap/lock.js.map +0 -1
  204. package/dist/devices-hap/meter.d.ts +0 -37
  205. package/dist/devices-hap/meter.d.ts.map +0 -1
  206. package/dist/devices-hap/meter.js +0 -379
  207. package/dist/devices-hap/meter.js.map +0 -1
  208. package/dist/devices-hap/meterplus.d.ts +0 -42
  209. package/dist/devices-hap/meterplus.d.ts.map +0 -1
  210. package/dist/devices-hap/meterplus.js +0 -384
  211. package/dist/devices-hap/meterplus.js.map +0 -1
  212. package/dist/devices-hap/meterpro.d.ts +0 -43
  213. package/dist/devices-hap/meterpro.d.ts.map +0 -1
  214. package/dist/devices-hap/meterpro.js +0 -468
  215. package/dist/devices-hap/meterpro.js.map +0 -1
  216. package/dist/devices-hap/motion.d.ts +0 -42
  217. package/dist/devices-hap/motion.d.ts.map +0 -1
  218. package/dist/devices-hap/motion.js +0 -345
  219. package/dist/devices-hap/motion.js.map +0 -1
  220. package/dist/devices-hap/plug.d.ts +0 -49
  221. package/dist/devices-hap/plug.d.ts.map +0 -1
  222. package/dist/devices-hap/plug.js +0 -395
  223. package/dist/devices-hap/plug.js.map +0 -1
  224. package/dist/devices-hap/relayswitch.d.ts +0 -96
  225. package/dist/devices-hap/relayswitch.d.ts.map +0 -1
  226. package/dist/devices-hap/relayswitch.js +0 -642
  227. package/dist/devices-hap/relayswitch.js.map +0 -1
  228. package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
  229. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
  230. package/dist/devices-hap/robotvacuumcleaner.js +0 -523
  231. package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
  232. package/dist/devices-hap/waterdetector.d.ts +0 -41
  233. package/dist/devices-hap/waterdetector.d.ts.map +0 -1
  234. package/dist/devices-hap/waterdetector.js +0 -356
  235. package/dist/devices-hap/waterdetector.js.map +0 -1
  236. package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -63
  237. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
  238. package/dist/devices-matter/BaseMatterAccessory.js +0 -100
  239. package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
  240. package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
  241. package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
  242. package/dist/devices-matter/ColorLightAccessory.js +0 -95
  243. package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
  244. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
  245. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
  246. package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -78
  247. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
  248. package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
  249. package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
  250. package/dist/devices-matter/ContactSensorAccessory.js +0 -34
  251. package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
  252. package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
  253. package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
  254. package/dist/devices-matter/DimmableLightAccessory.js +0 -167
  255. package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
  256. package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
  257. package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
  258. package/dist/devices-matter/DoorLockAccessory.js +0 -50
  259. package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
  260. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
  261. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
  262. package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -107
  263. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
  264. package/dist/devices-matter/FanAccessory.d.ts +0 -16
  265. package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
  266. package/dist/devices-matter/FanAccessory.js +0 -81
  267. package/dist/devices-matter/FanAccessory.js.map +0 -1
  268. package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
  269. package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
  270. package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
  271. package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
  272. package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
  273. package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
  274. package/dist/devices-matter/LeakSensorAccessory.js +0 -33
  275. package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
  276. package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
  277. package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
  278. package/dist/devices-matter/LightSensorAccessory.js +0 -34
  279. package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
  280. package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
  281. package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
  282. package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
  283. package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
  284. package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
  285. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
  286. package/dist/devices-matter/OnOffLightAccessory.js +0 -118
  287. package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
  288. package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -12
  289. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
  290. package/dist/devices-matter/OnOffOutletAccessory.js +0 -40
  291. package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
  292. package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
  293. package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
  294. package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
  295. package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
  296. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -68
  297. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
  298. package/dist/devices-matter/RoboticVacuumAccessory.js +0 -334
  299. package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
  300. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
  301. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
  302. package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
  303. package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
  304. package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
  305. package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
  306. package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
  307. package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
  308. package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
  309. package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
  310. package/dist/devices-matter/ThermostatAccessory.js +0 -95
  311. package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
  312. package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
  313. package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
  314. package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
  315. package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
  316. package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
  317. package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
  318. package/dist/devices-matter/WindowBlindAccessory.js +0 -80
  319. package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
  320. package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
  321. package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
  322. package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
  323. package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
  324. package/dist/devices-matter/custom/index.d.ts +0 -8
  325. package/dist/devices-matter/custom/index.d.ts.map +0 -1
  326. package/dist/devices-matter/custom/index.js +0 -8
  327. package/dist/devices-matter/custom/index.js.map +0 -1
  328. package/dist/devices-matter/index.d.ts +0 -29
  329. package/dist/devices-matter/index.d.ts.map +0 -1
  330. package/dist/devices-matter/index.js +0 -28
  331. package/dist/devices-matter/index.js.map +0 -1
  332. package/dist/index.test.d.ts +0 -2
  333. package/dist/index.test.d.ts.map +0 -1
  334. package/dist/index.test.js +0 -14
  335. package/dist/index.test.js.map +0 -1
  336. package/dist/irdevice/airconditioner.d.ts +0 -61
  337. package/dist/irdevice/airconditioner.d.ts.map +0 -1
  338. package/dist/irdevice/airconditioner.js +0 -472
  339. package/dist/irdevice/airconditioner.js.map +0 -1
  340. package/dist/irdevice/airpurifier.d.ts +0 -50
  341. package/dist/irdevice/airpurifier.d.ts.map +0 -1
  342. package/dist/irdevice/airpurifier.js +0 -213
  343. package/dist/irdevice/airpurifier.js.map +0 -1
  344. package/dist/irdevice/camera.d.ts +0 -32
  345. package/dist/irdevice/camera.d.ts.map +0 -1
  346. package/dist/irdevice/camera.js +0 -107
  347. package/dist/irdevice/camera.js.map +0 -1
  348. package/dist/irdevice/fan.d.ts +0 -36
  349. package/dist/irdevice/fan.d.ts.map +0 -1
  350. package/dist/irdevice/fan.js +0 -200
  351. package/dist/irdevice/fan.js.map +0 -1
  352. package/dist/irdevice/irdevice.d.ts +0 -68
  353. package/dist/irdevice/irdevice.d.ts.map +0 -1
  354. package/dist/irdevice/irdevice.js +0 -298
  355. package/dist/irdevice/irdevice.js.map +0 -1
  356. package/dist/irdevice/light.d.ts +0 -36
  357. package/dist/irdevice/light.d.ts.map +0 -1
  358. package/dist/irdevice/light.js +0 -206
  359. package/dist/irdevice/light.js.map +0 -1
  360. package/dist/irdevice/other.d.ts +0 -57
  361. package/dist/irdevice/other.d.ts.map +0 -1
  362. package/dist/irdevice/other.js +0 -778
  363. package/dist/irdevice/other.js.map +0 -1
  364. package/dist/irdevice/tv.d.ts +0 -45
  365. package/dist/irdevice/tv.d.ts.map +0 -1
  366. package/dist/irdevice/tv.js +0 -327
  367. package/dist/irdevice/tv.js.map +0 -1
  368. package/dist/irdevice/vacuumcleaner.d.ts +0 -28
  369. package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
  370. package/dist/irdevice/vacuumcleaner.js +0 -104
  371. package/dist/irdevice/vacuumcleaner.js.map +0 -1
  372. package/dist/irdevice/waterheater.d.ts +0 -30
  373. package/dist/irdevice/waterheater.d.ts.map +0 -1
  374. package/dist/irdevice/waterheater.js +0 -105
  375. package/dist/irdevice/waterheater.js.map +0 -1
  376. package/dist/platform-hap.d.ts +0 -149
  377. package/dist/platform-hap.d.ts.map +0 -1
  378. package/dist/platform-hap.js +0 -2861
  379. package/dist/platform-hap.js.map +0 -1
  380. package/dist/platform-matter.d.ts +0 -120
  381. package/dist/platform-matter.d.ts.map +0 -1
  382. package/dist/platform-matter.js +0 -966
  383. package/dist/platform-matter.js.map +0 -1
  384. package/dist/verifyconfig.test.d.ts +0 -2
  385. package/dist/verifyconfig.test.d.ts.map +0 -1
  386. package/dist/verifyconfig.test.js +0 -167
  387. package/dist/verifyconfig.test.js.map +0 -1
  388. package/src/custom.d.ts +0 -7
  389. package/src/devices-hap/airpurifier.ts +0 -563
  390. package/src/devices-hap/blindtilt.ts +0 -1049
  391. package/src/devices-hap/bot.ts +0 -900
  392. package/src/devices-hap/ceilinglight.ts +0 -742
  393. package/src/devices-hap/colorbulb.ts +0 -904
  394. package/src/devices-hap/contact.ts +0 -457
  395. package/src/devices-hap/curtain.ts +0 -944
  396. package/src/devices-hap/device.ts +0 -811
  397. package/src/devices-hap/fan.ts +0 -711
  398. package/src/devices-hap/hub.ts +0 -439
  399. package/src/devices-hap/humidifier.ts +0 -669
  400. package/src/devices-hap/iosensor.ts +0 -427
  401. package/src/devices-hap/lightstrip.ts +0 -836
  402. package/src/devices-hap/lock.ts +0 -620
  403. package/src/devices-hap/meter.ts +0 -426
  404. package/src/devices-hap/meterplus.ts +0 -430
  405. package/src/devices-hap/meterpro.ts +0 -522
  406. package/src/devices-hap/motion.ts +0 -390
  407. package/src/devices-hap/plug.ts +0 -423
  408. package/src/devices-hap/relayswitch.ts +0 -727
  409. package/src/devices-hap/robotvacuumcleaner.ts +0 -568
  410. package/src/devices-hap/waterdetector.ts +0 -400
  411. package/src/devices-matter/BaseMatterAccessory.ts +0 -131
  412. package/src/devices-matter/ColorLightAccessory.ts +0 -110
  413. package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -92
  414. package/src/devices-matter/ContactSensorAccessory.ts +0 -41
  415. package/src/devices-matter/DimmableLightAccessory.ts +0 -192
  416. package/src/devices-matter/DoorLockAccessory.ts +0 -60
  417. package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -123
  418. package/src/devices-matter/FanAccessory.ts +0 -95
  419. package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
  420. package/src/devices-matter/LeakSensorAccessory.ts +0 -40
  421. package/src/devices-matter/LightSensorAccessory.ts +0 -41
  422. package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
  423. package/src/devices-matter/OnOffLightAccessory.ts +0 -133
  424. package/src/devices-matter/OnOffOutletAccessory.ts +0 -46
  425. package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
  426. package/src/devices-matter/RoboticVacuumAccessory.ts +0 -407
  427. package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
  428. package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
  429. package/src/devices-matter/ThermostatAccessory.ts +0 -110
  430. package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
  431. package/src/devices-matter/WindowBlindAccessory.ts +0 -92
  432. package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
  433. package/src/devices-matter/custom/index.ts +0 -8
  434. package/src/devices-matter/index.ts +0 -29
  435. package/src/index.test.ts +0 -19
  436. package/src/irdevice/airconditioner.ts +0 -533
  437. package/src/irdevice/airpurifier.ts +0 -252
  438. package/src/irdevice/camera.ts +0 -129
  439. package/src/irdevice/fan.ts +0 -226
  440. package/src/irdevice/irdevice.ts +0 -344
  441. package/src/irdevice/light.ts +0 -246
  442. package/src/irdevice/other.ts +0 -790
  443. package/src/irdevice/tv.ts +0 -378
  444. package/src/irdevice/vacuumcleaner.ts +0 -126
  445. package/src/irdevice/waterheater.ts +0 -127
  446. package/src/platform-hap.ts +0 -2997
  447. package/src/platform-matter.ts +0 -1092
  448. package/src/verifyconfig.test.ts +0 -197
@@ -0,0 +1,36 @@
1
+ ---
2
+ name: E2E Verification Results
3
+ about: Submit manual E2E test results for homebridge-switchbot changes (Matter + node-switchbot v4)
4
+ title: "[E2E] <device-type> verification"
5
+ labels: e2e, testing
6
+ assignees: ''
7
+ ---
8
+
9
+ ### Environment
10
+ - Homebridge version:
11
+ - Plugin commit / branch:
12
+ - macOS version:
13
+ - Bluetooth available: (yes/no)
14
+ - openApiToken used: (yes/no)
15
+
16
+ ### Device under test
17
+ - Device type: (Light / Fan / Curtain / Lock)
18
+ - Device model:
19
+ - Connection used: (BLE / OpenAPI)
20
+
21
+ ### Steps performed
22
+ - Start Homebridge with plugin linked/built
23
+ - Verified Matter registration: (yes/no)
24
+ - Verified HAP fallback when Matter unavailable: (yes/no)
25
+ - Operations tested: (list)
26
+
27
+ ### Results
28
+ - Observed behavior (brief):
29
+ - Any errors in plugin logs (paste relevant snippets):
30
+
31
+ ### Verdict
32
+ - Pass/Fail:
33
+ - Notes and follow-ups:
34
+
35
+ ### Attachments
36
+ - Logs, screenshots, videos
@@ -0,0 +1,32 @@
1
+ name: CI
2
+
3
+ on:
4
+ push:
5
+ branches: ["main", "latest", "beta-*"]
6
+ pull_request:
7
+ branches: ["main", "latest"]
8
+
9
+ jobs:
10
+ build-and-test:
11
+ runs-on: ubuntu-latest
12
+ strategy:
13
+ matrix:
14
+ node-version: [20, 22]
15
+
16
+ steps:
17
+ - uses: actions/checkout@v4
18
+
19
+ - name: Setup Node.js
20
+ uses: actions/setup-node@v4
21
+ with:
22
+ node-version: ${{ matrix.node-version }}
23
+ cache: 'npm'
24
+
25
+ - name: Install dependencies
26
+ run: npm ci --legacy-peer-deps
27
+
28
+ - name: Build
29
+ run: npm run build
30
+
31
+ - name: Run tests
32
+ run: npm run test
@@ -0,0 +1,115 @@
1
+ name: Manual E2E (build + optional E2E)
2
+
3
+ on:
4
+ workflow_dispatch:
5
+ inputs:
6
+ run_e2e:
7
+ description: 'Set to true to run the E2E scripts (requires network-accessible Homebridge and valid tokens)'
8
+ required: false
9
+ default: 'false'
10
+ hb_url:
11
+ description: 'Homebridge UI URL (e.g. http://host:8581)'
12
+ required: false
13
+ hb_token:
14
+ description: 'Homebridge UI token (store as secret or pass here)'
15
+ required: false
16
+ light_accessory_id:
17
+ description: 'Accessory ID for light (when running E2E)'
18
+ required: false
19
+ fan_accessory_id:
20
+ description: 'Accessory ID for fan (when running E2E)'
21
+ required: false
22
+ curtain_accessory_id:
23
+ description: 'Accessory ID for curtain (when running E2E)'
24
+ required: false
25
+ lock_accessory_id:
26
+ description: 'Accessory ID for lock (when running E2E)'
27
+ required: false
28
+ light_accessory_name:
29
+ description: 'Accessory display name for light (optional, used to resolve ID)'
30
+ required: false
31
+ fan_accessory_name:
32
+ description: 'Accessory display name for fan (optional, used to resolve ID)'
33
+ required: false
34
+ curtain_accessory_name:
35
+ description: 'Accessory display name for curtain (optional, used to resolve ID)'
36
+ required: false
37
+ lock_accessory_name:
38
+ description: 'Accessory display name for lock (optional, used to resolve ID)'
39
+ required: false
40
+
41
+ jobs:
42
+ build-and-test:
43
+ runs-on: ubuntu-latest
44
+ steps:
45
+ - name: Checkout
46
+ uses: actions/checkout@v4
47
+
48
+ - name: Setup Node
49
+ uses: actions/setup-node@v4
50
+ with:
51
+ node-version: '18'
52
+
53
+ - name: Install dependencies
54
+ run: npm ci --prefer-offline --no-audit --progress=false
55
+
56
+ - name: Build
57
+ run: npm run build
58
+
59
+ - name: Run unit tests
60
+ run: npm run test
61
+
62
+ - name: "Optional: Run E2E scripts"
63
+ if: ${{ github.event.inputs.run_e2e == 'true' }}
64
+ env:
65
+ HB_URL: ${{ github.event.inputs.hb_url }}
66
+ HB_TOKEN: ${{ github.event.inputs.hb_token }}
67
+ LIGHT_ACCESSORY_ID: ${{ github.event.inputs.light_accessory_id }}
68
+ FAN_ACCESSORY_ID: ${{ github.event.inputs.fan_accessory_id }}
69
+ CURTAIN_ACCESSORY_ID: ${{ github.event.inputs.curtain_accessory_id }}
70
+ LOCK_ACCESSORY_ID: ${{ github.event.inputs.lock_accessory_id }}
71
+ LIGHT_ACCESSORY_NAME: ${{ github.event.inputs.light_accessory_name }}
72
+ FAN_ACCESSORY_NAME: ${{ github.event.inputs.fan_accessory_name }}
73
+ CURTAIN_ACCESSORY_NAME: ${{ github.event.inputs.curtain_accessory_name }}
74
+ LOCK_ACCESSORY_NAME: ${{ github.event.inputs.lock_accessory_name }}
75
+ run: |
76
+ echo "Running E2E scripts on runner ($HB_URL) - ensure the runner can access Homebridge and devices"
77
+ # resolve accessory IDs from names when IDs not provided
78
+ echo "Resolving accessory IDs from names (if provided)..."
79
+ rm -f resolved_ids.env || true
80
+ touch resolved_ids.env
81
+
82
+ resolve() {
83
+ idvar="$1"
84
+ namevar="$2"
85
+ nameval="$(eval echo \"\$$namevar\")"
86
+ idval="$(eval echo \"\$$idvar\")"
87
+ if [[ -z "$idval" && -n "$nameval" ]]; then
88
+ resp=$(curl -sS -H "Authorization: Bearer ${HB_TOKEN}" "${HB_URL}/api/accessories" || true)
89
+ found=$(printf '%s' "$resp" | node -e "const fs=require('fs');const name=process.argv[1];try{const j=JSON.parse(fs.readFileSync(0,'utf8'));const arr=Array.isArray(j)?j:(j.accessories||j)||[];for(const a of arr){if(a.displayName===name||a.name===name||(a.context&&a.context.deviceId===name)){if(a.aid){console.log(a.aid);process.exit(0);}else if(a.UUID){console.log(a.UUID);process.exit(0);}}} }catch(e){}" "$nameval" )
90
+ if [[ -n "$found" ]]; then
91
+ echo "Resolved $idvar=$found"
92
+ echo "$idvar=$found" >> resolved_ids.env
93
+ export "$idvar"="$found"
94
+ else
95
+ echo "Could not resolve $namevar='$nameval' to an id"
96
+ fi
97
+ fi
98
+ }
99
+
100
+ resolve LIGHT_ACCESSORY_ID LIGHT_ACCESSORY_NAME
101
+ resolve FAN_ACCESSORY_ID FAN_ACCESSORY_NAME
102
+ resolve CURTAIN_ACCESSORY_ID CURTAIN_ACCESSORY_NAME
103
+ resolve LOCK_ACCESSORY_ID LOCK_ACCESSORY_NAME
104
+
105
+ # set RUN_E2E so the conditional Vitest harness executes
106
+ export RUN_E2E=true
107
+ npm run test
108
+
109
+ - name: Upload logs (if any)
110
+ if: always()
111
+ uses: actions/upload-artifact@v4
112
+ with:
113
+ name: test-logs
114
+ path: |
115
+ test-output.log
@@ -22,8 +22,6 @@ jobs:
22
22
  with:
23
23
  release_type: ${{ needs.determine-release-type.outputs.release_type }}
24
24
  is_esm: ${{ needs.determine-release-type.outputs.is_esm == 'true' }}
25
- secrets:
26
- NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
27
25
 
28
26
  # 3️⃣ Publish to NPM and create GitHub release
29
27
  publish-release:
@@ -48,8 +46,6 @@ jobs:
48
46
  branch_name: ${{ needs.determine-release-type.outputs.branch_name }}
49
47
  release_type: ${{ needs.determine-release-type.outputs.release_type }}
50
48
  is_esm: ${{ needs.determine-release-type.outputs.is_esm == 'true' }}
51
- secrets:
52
- NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
53
49
 
54
50
  # 5️⃣ Notify if any previous job fails
55
51
  workflow-failure:
package/CHANGELOG.md CHANGED
@@ -1,7 +1,42 @@
1
1
  # Changelog
2
2
 
3
+ ## Unreleased — beta-4.3.3
4
+
5
+ ### Added
6
+ - Matter support (child-bridge) with Matter-first registration and HAP fallback.
7
+ - Hybrid `node-switchbot@4` client adapter (BLE + OpenAPI) with OpenAPI fallback.
8
+ - Manual and automated E2E scripts for lights, fans, curtains, and locks (`scripts/e2e/*`).
9
+ - Conditional Vitest E2E harness (`RUN_E2E=true`) and a manual GitHub Actions workflow to run E2E.
10
+
11
+ ### Changed
12
+ - Centralized Matter cluster/attribute numeric ID maps in `src/utils.ts`.
13
+ - Device descriptors refactored to use canonical Matter IDs.
14
+ - OpenAPI fallback hardened with timeouts, retries, backoff, and per-device retry limits.
15
+
16
+ ### Tests
17
+ - Added integration tests and a Matter test harness under `test/`.
18
+
19
+ ### Notes
20
+ - This is a beta release; follow migration notes in MIGRATION.md before upgrading.
21
+ # Changelog
22
+
3
23
  All notable changes to this project will be documented in this file. This project uses [Semantic Versioning](https://semver.org/)
4
24
 
25
+ ## [Unreleased]
26
+
27
+ ### What's Changed
28
+ - Matter platform: register only devices discovered via the SwitchBot OpenAPI by default. Per-device config overrides (by deviceId) are correctly merged into discovered devices.
29
+ - Add `options.allowConfigOnlyDevices` (boolean) to opt-in to registering devices that exist only in config (not discovered).
30
+ - Add unit tests for per-device merging and config-only behavior.
31
+ - Matter (Matter platform): default to discovery-first registration and automatically remove previously-registered "stale" accessories that are no longer discovered or configured.
32
+ - Add `options.keepStaleAccessories` (Advanced Settings) — opt-in to preserve previously-registered accessories when set to true. Default: false (stale accessories are removed automatically).
33
+ - Implement per-device OpenAPI polling with tracked timers and proper lifecycle cleanup (timers and BLE handlers cleared on unregister/shutdown).
34
+ - Centralize OpenAPI/BLE -> Matter mapping and expand BLE/OpenAPI parsing to include PM2.5/PM10/VOC/CO2, temperature/humidity, improved color parsing, and additional sensor/robot-vacuum fields.
35
+ - Prefer accessory-specific update helpers where available; fall back to generic Matter updates otherwise.
36
+ - Add unit tests covering BLE parsing, stale-accessory behavior, mapping helpers, and lifecycle cleanup.
37
+ - Add `options.dailyApiResetLocalMidnight` (boolean) to control whether the daily API budget resets at local midnight (true) or UTC midnight (false, default).
38
+ - Fix configuration schema placement for the "Reset Daily Counter at Local Midnight" option: remove erroneous insertion inside IR remote type list and expose it correctly under platform `options`.
39
+
5
40
  ## [4.3.1](https://github.com/OpenWonderLabs/homebridge-switchbot/releases/tag/v4.3.1) (2025-03-04)
6
41
 
7
42
  # *No New Releases During Lent*
@@ -0,0 +1,121 @@
1
+ # Manual End-to-End Verification
2
+
3
+ This guide walks through manual end-to-end verification for representative devices (light, fan, curtain, lock) across BLE and OpenAPI paths, and verifies Matter-first behavior with HAP fallback.
4
+
5
+ ## Prerequisites
6
+
7
+ - macOS with Bluetooth (for BLE verification)
8
+ - Local development environment set up (Node.js, npm)
9
+ - Homebridge installed locally or available (for running a test instance)
10
+ - SwitchBot devices available (at least one light, one fan, one curtain, one lock) OR corresponding cloud-hub access
11
+ - `openApiToken` when testing OpenAPI path
12
+
13
+ ## Quick setup
14
+
15
+ 1. Install dependencies:
16
+
17
+ ```bash
18
+ npm install --legacy-peer-deps
19
+ ```
20
+
21
+ 2. Build the plugin:
22
+
23
+ ```bash
24
+ npm run build
25
+ ```
26
+
27
+ 3. Recommended: run the watch task which links the plugin and restarts on changes (requires a local Homebridge installation):
28
+
29
+ ```bash
30
+ npm run watch
31
+ ```
32
+
33
+ Alternative (manual linking):
34
+
35
+ ```bash
36
+ npm run build && npm link
37
+ # then run your Homebridge instance (system/homebridge UI) so it picks up the linked plugin
38
+ ```
39
+
40
+ ## Example platform config snippets
41
+
42
+ OpenAPI (cloud) path — enable Matter preference:
43
+
44
+ ```json
45
+ {
46
+ "platform": "SwitchBot",
47
+ "openApiToken": "<YOUR_OPENAPI_TOKEN>",
48
+ "enableMatter": true,
49
+ "preferMatter": true
50
+ }
51
+ ```
52
+
53
+ BLE (direct) path — specify device by MAC address and prefer Matter:
54
+
55
+ ```json
56
+ {
57
+ "platform": "SwitchBot",
58
+ "devices": [
59
+ { "deviceId": "AA:BB:CC:DD:EE:FF", "type": "Light" }
60
+ ],
61
+ "enableMatter": true,
62
+ "preferMatter": true
63
+ }
64
+ ```
65
+
66
+ ## Verification checklist
67
+
68
+ Run the steps below for each device type; repeat once using BLE and once using OpenAPI (where applicable).
69
+
70
+ - Start Homebridge (with the plugin linked) and verify the plugin logs indicate Matter registration attempt.
71
+ - Look for log lines: `Attempting Matter registration for ...` or `Homebridge Matter API not available; will fallback to HAP`.
72
+ - Confirm accessories are registered in Matter (if Homebridge Matter child-bridge present) or appear as HAP accessories otherwise.
73
+
74
+ Device-specific tests
75
+
76
+ - Light
77
+ - Toggle On/Off from Home app (or Homebridge UI) and confirm device responds.
78
+ - Set brightness to 25%, 50%, 100% and confirm device reports matching levels.
79
+ - If supported, change color temperature and/or color; confirm device applies new values.
80
+
81
+ - Fan
82
+ - Toggle On/Off and verify.
83
+ - Change rotation speed in steps (e.g., 25%, 50%, 100%).
84
+ - Toggle oscillation / swing and verify position changes.
85
+
86
+ - Curtain
87
+ - Open and Close via Home app / controller and verify movement.
88
+ - Set specific position (e.g., 50%) and verify accurate reporting.
89
+
90
+ - Lock
91
+ - Lock and Unlock from controller and verify state.
92
+ - If lock user management supported, add/remove a test user per plugin UI and verify expected behavior.
93
+
94
+ ## Logs & debugging
95
+
96
+ - Enable verbose logs from Homebridge or the plugin. Example: start Homebridge with a higher log level, or watch console output from `npm run watch`.
97
+ - Capture logs while you perform each operation; note the request path used by the plugin:
98
+ - For OpenAPI: logs will show `https://api.switch-bot.com/v1.0/...` calls.
99
+ - For BLE: logs will show local device discovery / BLE connect attempts.
100
+
101
+ ## Run validation script (optional)
102
+
103
+ You can run the TypeScript build and unit tests to confirm the codebase is healthy before manual tests:
104
+
105
+ ```bash
106
+ npm run build && npm run test
107
+ ```
108
+
109
+ ## Expected outcomes
110
+
111
+ - Matter-first registration: when Homebridge Matter API is available and `enableMatter` is true, accessories should be registered as Matter child-bridge accessories and re-used if restored from cache.
112
+ - HAP fallback: when Matter API is not available, the plugin must continue to register HAP accessories and reuse restored accessories by `accessory.context.deviceId`.
113
+ - Hybrid client paths: when `openApiToken` present the plugin should prefer OpenAPI for devices reachable via cloud; when BLE is available and configured the plugin should connect directly via BLE.
114
+ - Logs should show robust retry attempts for transient network errors and per-device retry throttling if configured.
115
+
116
+ ## Next steps I can take
117
+
118
+ - Create small automated verification scripts that use the Homebridge API test harness to toggle characteristic values (requires more test harness code).
119
+ - Generate a printable checklist or a GitHub issue template for manual verification results.
120
+
121
+ If you want, I can add the automated verification scripts now — would you prefer a simple script that exercises On/Off and Brightness for lights, or a broader script covering all device types?
package/MIGRATION.md ADDED
@@ -0,0 +1,44 @@
1
+ # Migration notes — v4.3.x → beta
2
+
3
+ This document highlights important changes and recommended actions before upgrading to the beta containing Matter and `node-switchbot@4` support.
4
+
5
+ 1. Matter-first behavior
6
+ - The plugin will prefer registering accessories with Homebridge's Matter child-bridge when `enableMatter: true` and the Homebridge Matter API is available.
7
+ - If Matter is not available, the plugin falls back to HAP and will reuse restored accessories by `accessory.context.deviceId`.
8
+
9
+ 2. Configuration keys
10
+ - `enableMatter` (boolean) — enable Matter child-bridge registration.
11
+ - `preferMatter` (boolean) — prefer Matter for devices that support Matter descriptors; HAP fallback still available.
12
+ - `openApiToken` (string) — when present the plugin may prefer OpenAPI calls for cloud-reachable devices.
13
+ - `perDeviceMaxRetries`, `requestTimeout`, `maxRetries` — network retry/timing options for OpenAPI fallback.
14
+
15
+ - `writeDebounceMs` (number, milliseconds, default 100) — global write coalescing debounce window. Commands sent to the same device within this window are coalesced (last-write-wins) to reduce duplicate network/API commands. Set to `0` to disable coalescing if you require immediate, per-command delivery.
16
+
17
+ 3. Hybrid client
18
+ - The plugin dynamically imports `node-switchbot` when available and falls back to OpenAPI when an `openApiToken` is configured.
19
+ - If you rely on BLE-only operation, ensure devices and the host have BLE available.
20
+
21
+ 4. UI changes
22
+ - A small plugin UI was added; it's only enabled/copied into `dist/homebridge-ui` when `npm run build` is run and the UI files exist in `src/homebridge-ui/public`.
23
+
24
+ - The UI is always served when Homebridge UI support is present; there is no opt-out flag in the platform config. The server-side guard that previously allowed disabling the UI has been removed—if you need to restrict access, manage it via Homebridge UI / host network access controls.
25
+
26
+ 5. Tests and verification
27
+ - Manual E2E scripts are provided in `scripts/e2e/` and a GitHub `workflow_dispatch` job can optionally run them against a reachable Homebridge.
28
+ - Run the local test suite before upgrading to confirm TypeScript and unit tests pass: `npm run build && npm run test`.
29
+
30
+ 6. Rollback plan
31
+ - If the beta causes issues, revert to the previous stable plugin version by reinstalling the prior package or checking out the stable branch.
32
+
33
+ 7. Regenerating Matter ID maps
34
+
35
+ - A helper script `scripts/generate-matter-maps.js` is included to generate `src/matter-maps.generated.ts` from official zap/matter JSON metadata. The script expects a JSON input with a `clusters` array and will write mapped `MATTER_CLUSTER_IDS` and `MATTER_ATTRIBUTE_IDS` constants.
36
+ - Usage example:
37
+
38
+ ```bash
39
+ node scripts/generate-matter-maps.js ./zap-matter.json
40
+ ```
41
+
42
+ Place the official metadata JSON as `zap-matter.json` at the repo root (or pass a path) and commit the generated `src/matter-maps.generated.ts` to keep maps up to date.
43
+
44
+ If you want, I can open a PR with these notes and the changelog stub targeting a beta branch I create next.
package/README.md CHANGED
@@ -52,16 +52,13 @@
52
52
  ## Troubleshooting
53
53
 
54
54
  - ### If using Linux / Raspberry Pi OS
55
-
56
55
  1. `bluetoothctl` must be installed on the device, otherwise it cannot communicate via Bluetooth. Enable it with `sudo bluetoothctl power on`.
57
56
 
58
57
  2. If errors occur, while enabling it, restart the process:
59
-
60
58
  - `rfkill block bluetooth`
61
59
  - `rfkill unblock bluetooth`
62
60
 
63
61
  3. Also make sure, that the computer can discover the SwitchBot device:
64
-
65
62
  - `sudo bluetoothctl`
66
63
  - `scan on`
67
64
 
@@ -218,6 +215,62 @@
218
215
  - Supports Fan Device Type
219
216
  - Allows for On/Off Controls
220
217
 
218
+ ## Matter Platform
219
+
220
+ ### Batched refresh and API load control
221
+
222
+ By default, the Matter platform uses a single batched refresh to update device status. You can tune or override this behavior with the following options under `options`:
223
+
224
+ - `matterBatchEnabled` (boolean, default true): enable/disable platform-level batched refresh. Devices with a per-device `refreshRate` still run their own timers.
225
+ - `matterBatchRefreshRate` (number, seconds): batch interval (falls back to `options.refreshRate`, then 300 if not set).
226
+ - `matterBatchConcurrency` (number): limit of parallel OpenAPI status calls during a batch (default 5).
227
+ - `matterBatchJitter` (number, seconds): random startup delay before the first batch to reduce synchronized spikes.
228
+
229
+ Device-level override:
230
+
231
+ - If a device sets `refreshRate` in its config, it uses a per-device timer and is excluded from the platform batch.
232
+
233
+ Reliability and rate-limiting:
234
+
235
+ - Each device status call retries with exponential backoff on non-success responses.
236
+ - After exhausting retries, the device enters a short cooldown before being retried; cooldowns are persisted across restarts.
237
+ - The batch worklist is randomized every cycle to further distribute API load.
238
+
239
+ These controls keep API usage smooth and predictable while preserving per-device control when needed.
240
+
241
+ ## What's new in the v4 beta (summary)
242
+
243
+ - Matter-first: when Homebridge Matter is available the plugin now prefers registering Matter accessories (with HAP fallback).
244
+ - Hybrid client: the plugin dynamically imports `node-switchbot@4` if available and falls back to OpenAPI when `openApiToken` is configured.
245
+ - UI always served: the plugin UI is packaged into `dist/homebridge-ui` and is always served when Homebridge UI support is present; there is no platform-level opt-out.
246
+ - OpenAPI hardening: OpenAPI calls have AbortController timeouts, jittered exponential backoff, per-device retry limits and cooldowns, and safe response parsing for resilient behavior.
247
+
248
+ - Write coalescing (debounce): command writes to the same device are coalesced by default to avoid command floods. Configure with `writeDebounceMs` (milliseconds, default 100). Set to `0` to disable coalescing.
249
+
250
+ See `MIGRATION.md` for migration notes and recommended upgrade steps.
251
+
252
+ ## OpenAPI rate limiting and daily budget
253
+
254
+ To prevent hitting SwitchBot’s daily OpenAPI limit, the plugin provides several platform-level options under `options`:
255
+
256
+ - `dailyApiLimit` (number, default 10000): maximum OpenAPI requests per day that the plugin will allow.
257
+ - `dailyApiReserveForCommands` (number, default 1000): requests reserved for user actions so background polling pauses before the hard limit.
258
+ - `webhookOnlyOnReserve` (boolean, default false): when remaining budget reaches the reserve, pause background polling/discovery; webhooks and commands continue until the hard limit.
259
+ - `dailyApiResetLocalMidnight` (boolean, default false): if true, resets the daily counter at local midnight; when false, resets at UTC midnight.
260
+
261
+ Example (excerpt):
262
+
263
+ ```json
264
+ {
265
+ "options": {
266
+ "dailyApiLimit": 10000,
267
+ "dailyApiReserveForCommands": 1000,
268
+ "webhookOnlyOnReserve": false,
269
+ "dailyApiResetLocalMidnight": true
270
+ }
271
+ }
272
+ ```
273
+
221
274
  ## SwitchBot APIs
222
275
 
223
276
  - [OpenWonderLabs/node-switchbot](https://github.com/OpenWonderLabs/node-switchbot)