@switchbot/homebridge-switchbot 5.0.0-beta.8 → 5.0.0-beta.81

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 (447) 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 +35 -0
  5. package/E2E-VERIFICATION.md +121 -0
  6. package/MIGRATION.md +44 -0
  7. package/README.md +56 -3
  8. package/config.schema.json +83 -14807
  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 +580 -246
  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 +286 -35
  25. package/dist/homebridge-ui/server.js.map +1 -1
  26. package/dist/index.d.ts.map +1 -1
  27. package/dist/index.js +4 -11
  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 +411 -0
  32. package/dist/platform.js.map +1 -0
  33. package/dist/settings.d.ts +10 -249
  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 +194 -0
  40. package/dist/switchbotClient.js.map +1 -0
  41. package/dist/utils.d.ts +39 -50
  42. package/dist/utils.d.ts.map +1 -1
  43. package/dist/utils.js +39 -688
  44. package/dist/utils.js.map +1 -1
  45. package/docs/assets/highlight.css +14 -0
  46. package/docs/assets/icons.js +1 -1
  47. package/docs/assets/icons.svg +1 -1
  48. package/docs/assets/main.js +2 -2
  49. package/docs/assets/style.css +3 -3
  50. package/docs/index.html +77 -13
  51. package/docs/variables/default.html +1 -1
  52. package/eslint.config.js +2 -8
  53. package/package.json +21 -28
  54. package/scripts/e2e/README.md +25 -0
  55. package/scripts/e2e/curtain-e2e.sh +70 -0
  56. package/scripts/e2e/fan-e2e.sh +75 -0
  57. package/scripts/e2e/light-advanced-e2e.sh +97 -0
  58. package/scripts/e2e/light-e2e.sh +75 -0
  59. package/scripts/e2e/list-accessories.sh +19 -0
  60. package/scripts/e2e/lock-e2e.sh +65 -0
  61. package/scripts/generate-matter-maps.js +60 -0
  62. package/scripts/run-e2e-local.sh +14 -0
  63. package/src/deviceFactory.ts +122 -0
  64. package/src/devices/deviceBase.ts +141 -0
  65. package/src/devices/genericDevice.ts +965 -0
  66. package/src/homebridge-ui/public/index.html +580 -246
  67. package/src/homebridge-ui/server.ts +340 -40
  68. package/src/index.ts +4 -12
  69. package/src/platform.ts +407 -0
  70. package/src/settings.ts +12 -277
  71. package/src/switchbotClient.ts +203 -0
  72. package/src/utils.ts +45 -713
  73. package/test/accessory-restore.spec.ts +73 -0
  74. package/test/device-mapping.spec.ts +37 -0
  75. package/test/deviceFactory.spec.ts +18 -0
  76. package/test/e2e/run-e2e.spec.ts +50 -0
  77. package/test/fan-swing.spec.ts +29 -0
  78. package/test/helpers/matter-harness.ts +53 -0
  79. package/test/lock-users.spec.ts +44 -0
  80. package/test/matter-childbridge.spec.ts +55 -0
  81. package/test/matter-descriptors.spec.ts +97 -0
  82. package/test/matter-device-state.spec.ts +101 -0
  83. package/test/matter-integration.spec.ts +70 -0
  84. package/test/platform.integration.spec.ts +55 -0
  85. package/test/switchbot-client-debounce.spec.ts +131 -0
  86. package/test/switchbot-client-openapi.spec.ts +56 -0
  87. package/test/switchbotClient.spec.ts +10 -0
  88. package/test/utils.spec.ts +20 -0
  89. package/vitest.config.ts +7 -0
  90. package/coverage/base.css +0 -224
  91. package/coverage/block-navigation.js +0 -87
  92. package/coverage/clover.xml +0 -15847
  93. package/coverage/coverage-final.json +0 -42
  94. package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
  95. package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
  96. package/coverage/docs/assets/dmt/index.html +0 -131
  97. package/coverage/docs/assets/hierarchy.js.html +0 -85
  98. package/coverage/docs/assets/icons.js.html +0 -136
  99. package/coverage/docs/assets/index.html +0 -146
  100. package/coverage/docs/assets/main.js.html +0 -265
  101. package/coverage/favicon.png +0 -0
  102. package/coverage/index.html +0 -191
  103. package/coverage/prettify.css +0 -1
  104. package/coverage/prettify.js +0 -2
  105. package/coverage/sort-arrow-sprite.png +0 -0
  106. package/coverage/sorter.js +0 -196
  107. package/coverage/src/device/blindtilt.ts.html +0 -3238
  108. package/coverage/src/device/bot.ts.html +0 -2803
  109. package/coverage/src/device/ceilinglight.ts.html +0 -2338
  110. package/coverage/src/device/colorbulb.ts.html +0 -2824
  111. package/coverage/src/device/contact.ts.html +0 -1465
  112. package/coverage/src/device/curtain.ts.html +0 -2869
  113. package/coverage/src/device/device.ts.html +0 -2500
  114. package/coverage/src/device/fan.ts.html +0 -2242
  115. package/coverage/src/device/hub.ts.html +0 -1408
  116. package/coverage/src/device/humidifier.ts.html +0 -2116
  117. package/coverage/src/device/index.html +0 -416
  118. package/coverage/src/device/iosensor.ts.html +0 -1375
  119. package/coverage/src/device/lightstrip.ts.html +0 -2617
  120. package/coverage/src/device/lock.ts.html +0 -1963
  121. package/coverage/src/device/meter.ts.html +0 -1372
  122. package/coverage/src/device/meterplus.ts.html +0 -1384
  123. package/coverage/src/device/meterpro.ts.html +0 -1618
  124. package/coverage/src/device/motion.ts.html +0 -1264
  125. package/coverage/src/device/plug.ts.html +0 -1372
  126. package/coverage/src/device/relayswitch.ts.html +0 -2284
  127. package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
  128. package/coverage/src/device/waterdetector.ts.html +0 -1294
  129. package/coverage/src/homebridge-ui/index.html +0 -116
  130. package/coverage/src/homebridge-ui/server.ts.html +0 -229
  131. package/coverage/src/index.html +0 -161
  132. package/coverage/src/index.ts.html +0 -124
  133. package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
  134. package/coverage/src/irdevice/airpurifier.ts.html +0 -844
  135. package/coverage/src/irdevice/camera.ts.html +0 -475
  136. package/coverage/src/irdevice/fan.ts.html +0 -766
  137. package/coverage/src/irdevice/index.html +0 -251
  138. package/coverage/src/irdevice/irdevice.ts.html +0 -1117
  139. package/coverage/src/irdevice/light.ts.html +0 -826
  140. package/coverage/src/irdevice/other.ts.html +0 -2458
  141. package/coverage/src/irdevice/tv.ts.html +0 -1222
  142. package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
  143. package/coverage/src/irdevice/waterheater.ts.html +0 -469
  144. package/coverage/src/platform.ts.html +0 -8776
  145. package/coverage/src/settings.ts.html +0 -934
  146. package/coverage/src/utils.ts.html +0 -2092
  147. package/dist/devices-hap/airpurifier.d.ts +0 -54
  148. package/dist/devices-hap/airpurifier.d.ts.map +0 -1
  149. package/dist/devices-hap/airpurifier.js +0 -527
  150. package/dist/devices-hap/airpurifier.js.map +0 -1
  151. package/dist/devices-hap/blindtilt.d.ts +0 -90
  152. package/dist/devices-hap/blindtilt.d.ts.map +0 -1
  153. package/dist/devices-hap/blindtilt.js +0 -974
  154. package/dist/devices-hap/blindtilt.js.map +0 -1
  155. package/dist/devices-hap/bot.d.ts +0 -102
  156. package/dist/devices-hap/bot.d.ts.map +0 -1
  157. package/dist/devices-hap/bot.js +0 -811
  158. package/dist/devices-hap/bot.js.map +0 -1
  159. package/dist/devices-hap/ceilinglight.d.ts +0 -85
  160. package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
  161. package/dist/devices-hap/ceilinglight.js +0 -701
  162. package/dist/devices-hap/ceilinglight.js.map +0 -1
  163. package/dist/devices-hap/colorbulb.d.ts +0 -88
  164. package/dist/devices-hap/colorbulb.d.ts.map +0 -1
  165. package/dist/devices-hap/colorbulb.js +0 -881
  166. package/dist/devices-hap/colorbulb.js.map +0 -1
  167. package/dist/devices-hap/contact.d.ts +0 -44
  168. package/dist/devices-hap/contact.d.ts.map +0 -1
  169. package/dist/devices-hap/contact.js +0 -409
  170. package/dist/devices-hap/contact.js.map +0 -1
  171. package/dist/devices-hap/curtain.d.ts +0 -73
  172. package/dist/devices-hap/curtain.d.ts.map +0 -1
  173. package/dist/devices-hap/curtain.js +0 -869
  174. package/dist/devices-hap/curtain.js.map +0 -1
  175. package/dist/devices-hap/device.d.ts +0 -98
  176. package/dist/devices-hap/device.d.ts.map +0 -1
  177. package/dist/devices-hap/device.js +0 -749
  178. package/dist/devices-hap/device.js.map +0 -1
  179. package/dist/devices-hap/fan.d.ts +0 -69
  180. package/dist/devices-hap/fan.d.ts.map +0 -1
  181. package/dist/devices-hap/fan.js +0 -649
  182. package/dist/devices-hap/fan.js.map +0 -1
  183. package/dist/devices-hap/hub.d.ts +0 -37
  184. package/dist/devices-hap/hub.d.ts.map +0 -1
  185. package/dist/devices-hap/hub.js +0 -392
  186. package/dist/devices-hap/hub.js.map +0 -1
  187. package/dist/devices-hap/humidifier.d.ts +0 -68
  188. package/dist/devices-hap/humidifier.d.ts.map +0 -1
  189. package/dist/devices-hap/humidifier.js +0 -628
  190. package/dist/devices-hap/humidifier.js.map +0 -1
  191. package/dist/devices-hap/iosensor.d.ts +0 -42
  192. package/dist/devices-hap/iosensor.d.ts.map +0 -1
  193. package/dist/devices-hap/iosensor.js +0 -382
  194. package/dist/devices-hap/iosensor.js.map +0 -1
  195. package/dist/devices-hap/lightstrip.d.ts +0 -79
  196. package/dist/devices-hap/lightstrip.d.ts.map +0 -1
  197. package/dist/devices-hap/lightstrip.js +0 -797
  198. package/dist/devices-hap/lightstrip.js.map +0 -1
  199. package/dist/devices-hap/lock.d.ts +0 -53
  200. package/dist/devices-hap/lock.d.ts.map +0 -1
  201. package/dist/devices-hap/lock.js +0 -561
  202. package/dist/devices-hap/lock.js.map +0 -1
  203. package/dist/devices-hap/meter.d.ts +0 -37
  204. package/dist/devices-hap/meter.d.ts.map +0 -1
  205. package/dist/devices-hap/meter.js +0 -379
  206. package/dist/devices-hap/meter.js.map +0 -1
  207. package/dist/devices-hap/meterplus.d.ts +0 -42
  208. package/dist/devices-hap/meterplus.d.ts.map +0 -1
  209. package/dist/devices-hap/meterplus.js +0 -384
  210. package/dist/devices-hap/meterplus.js.map +0 -1
  211. package/dist/devices-hap/meterpro.d.ts +0 -43
  212. package/dist/devices-hap/meterpro.d.ts.map +0 -1
  213. package/dist/devices-hap/meterpro.js +0 -468
  214. package/dist/devices-hap/meterpro.js.map +0 -1
  215. package/dist/devices-hap/motion.d.ts +0 -42
  216. package/dist/devices-hap/motion.d.ts.map +0 -1
  217. package/dist/devices-hap/motion.js +0 -345
  218. package/dist/devices-hap/motion.js.map +0 -1
  219. package/dist/devices-hap/plug.d.ts +0 -49
  220. package/dist/devices-hap/plug.d.ts.map +0 -1
  221. package/dist/devices-hap/plug.js +0 -395
  222. package/dist/devices-hap/plug.js.map +0 -1
  223. package/dist/devices-hap/relayswitch.d.ts +0 -96
  224. package/dist/devices-hap/relayswitch.d.ts.map +0 -1
  225. package/dist/devices-hap/relayswitch.js +0 -642
  226. package/dist/devices-hap/relayswitch.js.map +0 -1
  227. package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
  228. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
  229. package/dist/devices-hap/robotvacuumcleaner.js +0 -523
  230. package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
  231. package/dist/devices-hap/waterdetector.d.ts +0 -41
  232. package/dist/devices-hap/waterdetector.d.ts.map +0 -1
  233. package/dist/devices-hap/waterdetector.js +0 -356
  234. package/dist/devices-hap/waterdetector.js.map +0 -1
  235. package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -63
  236. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
  237. package/dist/devices-matter/BaseMatterAccessory.js +0 -100
  238. package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
  239. package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
  240. package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
  241. package/dist/devices-matter/ColorLightAccessory.js +0 -95
  242. package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
  243. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
  244. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
  245. package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -78
  246. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
  247. package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
  248. package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
  249. package/dist/devices-matter/ContactSensorAccessory.js +0 -34
  250. package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
  251. package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
  252. package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
  253. package/dist/devices-matter/DimmableLightAccessory.js +0 -167
  254. package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
  255. package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
  256. package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
  257. package/dist/devices-matter/DoorLockAccessory.js +0 -50
  258. package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
  259. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
  260. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
  261. package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -107
  262. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
  263. package/dist/devices-matter/FanAccessory.d.ts +0 -16
  264. package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
  265. package/dist/devices-matter/FanAccessory.js +0 -81
  266. package/dist/devices-matter/FanAccessory.js.map +0 -1
  267. package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
  268. package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
  269. package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
  270. package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
  271. package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
  272. package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
  273. package/dist/devices-matter/LeakSensorAccessory.js +0 -33
  274. package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
  275. package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
  276. package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
  277. package/dist/devices-matter/LightSensorAccessory.js +0 -34
  278. package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
  279. package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
  280. package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
  281. package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
  282. package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
  283. package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
  284. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
  285. package/dist/devices-matter/OnOffLightAccessory.js +0 -118
  286. package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
  287. package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -12
  288. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
  289. package/dist/devices-matter/OnOffOutletAccessory.js +0 -40
  290. package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
  291. package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
  292. package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
  293. package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
  294. package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
  295. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -68
  296. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
  297. package/dist/devices-matter/RoboticVacuumAccessory.js +0 -334
  298. package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
  299. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
  300. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
  301. package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
  302. package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
  303. package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
  304. package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
  305. package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
  306. package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
  307. package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
  308. package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
  309. package/dist/devices-matter/ThermostatAccessory.js +0 -95
  310. package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
  311. package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
  312. package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
  313. package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
  314. package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
  315. package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
  316. package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
  317. package/dist/devices-matter/WindowBlindAccessory.js +0 -80
  318. package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
  319. package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
  320. package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
  321. package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
  322. package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
  323. package/dist/devices-matter/custom/index.d.ts +0 -8
  324. package/dist/devices-matter/custom/index.d.ts.map +0 -1
  325. package/dist/devices-matter/custom/index.js +0 -8
  326. package/dist/devices-matter/custom/index.js.map +0 -1
  327. package/dist/devices-matter/index.d.ts +0 -29
  328. package/dist/devices-matter/index.d.ts.map +0 -1
  329. package/dist/devices-matter/index.js +0 -28
  330. package/dist/devices-matter/index.js.map +0 -1
  331. package/dist/index.test.d.ts +0 -2
  332. package/dist/index.test.d.ts.map +0 -1
  333. package/dist/index.test.js +0 -14
  334. package/dist/index.test.js.map +0 -1
  335. package/dist/irdevice/airconditioner.d.ts +0 -61
  336. package/dist/irdevice/airconditioner.d.ts.map +0 -1
  337. package/dist/irdevice/airconditioner.js +0 -472
  338. package/dist/irdevice/airconditioner.js.map +0 -1
  339. package/dist/irdevice/airpurifier.d.ts +0 -50
  340. package/dist/irdevice/airpurifier.d.ts.map +0 -1
  341. package/dist/irdevice/airpurifier.js +0 -213
  342. package/dist/irdevice/airpurifier.js.map +0 -1
  343. package/dist/irdevice/camera.d.ts +0 -32
  344. package/dist/irdevice/camera.d.ts.map +0 -1
  345. package/dist/irdevice/camera.js +0 -107
  346. package/dist/irdevice/camera.js.map +0 -1
  347. package/dist/irdevice/fan.d.ts +0 -36
  348. package/dist/irdevice/fan.d.ts.map +0 -1
  349. package/dist/irdevice/fan.js +0 -200
  350. package/dist/irdevice/fan.js.map +0 -1
  351. package/dist/irdevice/irdevice.d.ts +0 -68
  352. package/dist/irdevice/irdevice.d.ts.map +0 -1
  353. package/dist/irdevice/irdevice.js +0 -298
  354. package/dist/irdevice/irdevice.js.map +0 -1
  355. package/dist/irdevice/light.d.ts +0 -36
  356. package/dist/irdevice/light.d.ts.map +0 -1
  357. package/dist/irdevice/light.js +0 -206
  358. package/dist/irdevice/light.js.map +0 -1
  359. package/dist/irdevice/other.d.ts +0 -57
  360. package/dist/irdevice/other.d.ts.map +0 -1
  361. package/dist/irdevice/other.js +0 -778
  362. package/dist/irdevice/other.js.map +0 -1
  363. package/dist/irdevice/tv.d.ts +0 -45
  364. package/dist/irdevice/tv.d.ts.map +0 -1
  365. package/dist/irdevice/tv.js +0 -327
  366. package/dist/irdevice/tv.js.map +0 -1
  367. package/dist/irdevice/vacuumcleaner.d.ts +0 -28
  368. package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
  369. package/dist/irdevice/vacuumcleaner.js +0 -104
  370. package/dist/irdevice/vacuumcleaner.js.map +0 -1
  371. package/dist/irdevice/waterheater.d.ts +0 -30
  372. package/dist/irdevice/waterheater.d.ts.map +0 -1
  373. package/dist/irdevice/waterheater.js +0 -105
  374. package/dist/irdevice/waterheater.js.map +0 -1
  375. package/dist/platform-hap.d.ts +0 -149
  376. package/dist/platform-hap.d.ts.map +0 -1
  377. package/dist/platform-hap.js +0 -2861
  378. package/dist/platform-hap.js.map +0 -1
  379. package/dist/platform-matter.d.ts +0 -120
  380. package/dist/platform-matter.d.ts.map +0 -1
  381. package/dist/platform-matter.js +0 -966
  382. package/dist/platform-matter.js.map +0 -1
  383. package/dist/verifyconfig.test.d.ts +0 -2
  384. package/dist/verifyconfig.test.d.ts.map +0 -1
  385. package/dist/verifyconfig.test.js +0 -167
  386. package/dist/verifyconfig.test.js.map +0 -1
  387. package/src/custom.d.ts +0 -7
  388. package/src/devices-hap/airpurifier.ts +0 -563
  389. package/src/devices-hap/blindtilt.ts +0 -1049
  390. package/src/devices-hap/bot.ts +0 -900
  391. package/src/devices-hap/ceilinglight.ts +0 -742
  392. package/src/devices-hap/colorbulb.ts +0 -904
  393. package/src/devices-hap/contact.ts +0 -457
  394. package/src/devices-hap/curtain.ts +0 -944
  395. package/src/devices-hap/device.ts +0 -811
  396. package/src/devices-hap/fan.ts +0 -711
  397. package/src/devices-hap/hub.ts +0 -439
  398. package/src/devices-hap/humidifier.ts +0 -669
  399. package/src/devices-hap/iosensor.ts +0 -427
  400. package/src/devices-hap/lightstrip.ts +0 -836
  401. package/src/devices-hap/lock.ts +0 -620
  402. package/src/devices-hap/meter.ts +0 -426
  403. package/src/devices-hap/meterplus.ts +0 -430
  404. package/src/devices-hap/meterpro.ts +0 -522
  405. package/src/devices-hap/motion.ts +0 -390
  406. package/src/devices-hap/plug.ts +0 -423
  407. package/src/devices-hap/relayswitch.ts +0 -727
  408. package/src/devices-hap/robotvacuumcleaner.ts +0 -568
  409. package/src/devices-hap/waterdetector.ts +0 -400
  410. package/src/devices-matter/BaseMatterAccessory.ts +0 -131
  411. package/src/devices-matter/ColorLightAccessory.ts +0 -110
  412. package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -92
  413. package/src/devices-matter/ContactSensorAccessory.ts +0 -41
  414. package/src/devices-matter/DimmableLightAccessory.ts +0 -192
  415. package/src/devices-matter/DoorLockAccessory.ts +0 -60
  416. package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -123
  417. package/src/devices-matter/FanAccessory.ts +0 -95
  418. package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
  419. package/src/devices-matter/LeakSensorAccessory.ts +0 -40
  420. package/src/devices-matter/LightSensorAccessory.ts +0 -41
  421. package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
  422. package/src/devices-matter/OnOffLightAccessory.ts +0 -133
  423. package/src/devices-matter/OnOffOutletAccessory.ts +0 -46
  424. package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
  425. package/src/devices-matter/RoboticVacuumAccessory.ts +0 -407
  426. package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
  427. package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
  428. package/src/devices-matter/ThermostatAccessory.ts +0 -110
  429. package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
  430. package/src/devices-matter/WindowBlindAccessory.ts +0 -92
  431. package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
  432. package/src/devices-matter/custom/index.ts +0 -8
  433. package/src/devices-matter/index.ts +0 -29
  434. package/src/index.test.ts +0 -19
  435. package/src/irdevice/airconditioner.ts +0 -533
  436. package/src/irdevice/airpurifier.ts +0 -252
  437. package/src/irdevice/camera.ts +0 -129
  438. package/src/irdevice/fan.ts +0 -226
  439. package/src/irdevice/irdevice.ts +0 -344
  440. package/src/irdevice/light.ts +0 -246
  441. package/src/irdevice/other.ts +0 -790
  442. package/src/irdevice/tv.ts +0 -378
  443. package/src/irdevice/vacuumcleaner.ts +0 -126
  444. package/src/irdevice/waterheater.ts +0 -127
  445. package/src/platform-hap.ts +0 -2997
  446. package/src/platform-matter.ts +0 -1092
  447. 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,61 @@
1
+ name: CI
2
+
3
+ on:
4
+ push:
5
+ branches: ["latest", "beta-*", "main"]
6
+ pull_request:
7
+ branches: ["latest", "main"]
8
+
9
+ jobs:
10
+ test:
11
+ name: Build and Test
12
+ runs-on: ubuntu-latest
13
+ steps:
14
+ - name: Checkout
15
+ uses: actions/checkout@v4
16
+
17
+ - name: Use Node.js
18
+ uses: actions/setup-node@v4
19
+ with:
20
+ node-version: '20'
21
+
22
+ - name: Install dependencies
23
+ run: npm ci --legacy-peer-deps
24
+
25
+ - name: Build
26
+ run: npm run build
27
+
28
+ - name: Run tests
29
+ run: npm run test
30
+ name: CI
31
+
32
+ on:
33
+ push:
34
+ branches: [ main, latest ]
35
+ pull_request:
36
+ branches: [ main, latest ]
37
+
38
+ jobs:
39
+ build-and-test:
40
+ runs-on: ubuntu-latest
41
+ strategy:
42
+ matrix:
43
+ node-version: [20, 22]
44
+
45
+ steps:
46
+ - uses: actions/checkout@v4
47
+
48
+ - name: Setup Node.js
49
+ uses: actions/setup-node@v4
50
+ with:
51
+ node-version: ${{ matrix.node-version }}
52
+ cache: 'npm'
53
+
54
+ - name: Install dependencies
55
+ run: npm ci --legacy-peer-deps
56
+
57
+ - name: Build
58
+ run: npm run build
59
+
60
+ - name: Run tests
61
+ run: npm test
@@ -0,0 +1,103 @@
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
+
29
+ jobs:
30
+ build-and-test:
31
+ runs-on: ubuntu-latest
32
+ steps:
33
+ - name: Checkout
34
+ uses: actions/checkout@v4
35
+
36
+ - name: Setup Node
37
+ uses: actions/setup-node@v4
38
+ with:
39
+ node-version: '18'
40
+
41
+ - name: Install dependencies
42
+ run: npm ci --prefer-offline --no-audit --progress=false
43
+
44
+ - name: Build
45
+ run: npm run build
46
+
47
+ - name: Run unit tests
48
+ run: npm run test
49
+
50
+ - name: Optional: Run E2E scripts
51
+ if: ${{ github.event.inputs.run_e2e == 'true' }}
52
+ env:
53
+ HB_URL: ${{ github.event.inputs.hb_url }}
54
+ HB_TOKEN: ${{ github.event.inputs.hb_token }}
55
+ LIGHT_ACCESSORY_ID: ${{ github.event.inputs.light_accessory_id }}
56
+ FAN_ACCESSORY_ID: ${{ github.event.inputs.fan_accessory_id }}
57
+ CURTAIN_ACCESSORY_ID: ${{ github.event.inputs.curtain_accessory_id }}
58
+ LOCK_ACCESSORY_ID: ${{ github.event.inputs.lock_accessory_id }}
59
+ LIGHT_ACCESSORY_NAME: ${{ github.event.inputs.light_accessory_name }}
60
+ FAN_ACCESSORY_NAME: ${{ github.event.inputs.fan_accessory_name }}
61
+ CURTAIN_ACCESSORY_NAME: ${{ github.event.inputs.curtain_accessory_name }}
62
+ LOCK_ACCESSORY_NAME: ${{ github.event.inputs.lock_accessory_name }}
63
+ run: |
64
+ echo "Running E2E scripts on runner ($HB_URL) - ensure the runner can access Homebridge and devices"
65
+ # resolve accessory IDs from names when IDs not provided
66
+ echo "Resolving accessory IDs from names (if provided)..."
67
+ rm -f resolved_ids.env || true
68
+ touch resolved_ids.env
69
+
70
+ resolve() {
71
+ idvar="$1"
72
+ namevar="$2"
73
+ nameval="$(eval echo \"\$$namevar\")"
74
+ idval="$(eval echo \"\$$idvar\")"
75
+ if [[ -z "$idval" && -n "$nameval" ]]; then
76
+ resp=$(curl -sS -H "Authorization: Bearer ${HB_TOKEN}" "${HB_URL}/api/accessories" || true)
77
+ 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" )
78
+ if [[ -n "$found" ]]; then
79
+ echo "Resolved $idvar=$found"
80
+ echo "$idvar=$found" >> resolved_ids.env
81
+ export "$idvar"="$found"
82
+ else
83
+ echo "Could not resolve $namevar='$nameval' to an id"
84
+ fi
85
+ fi
86
+ }
87
+
88
+ resolve LIGHT_ACCESSORY_ID LIGHT_ACCESSORY_NAME
89
+ resolve FAN_ACCESSORY_ID FAN_ACCESSORY_NAME
90
+ resolve CURTAIN_ACCESSORY_ID CURTAIN_ACCESSORY_NAME
91
+ resolve LOCK_ACCESSORY_ID LOCK_ACCESSORY_NAME
92
+
93
+ # set RUN_E2E so the conditional Vitest harness executes
94
+ export RUN_E2E=true
95
+ npm run test
96
+
97
+ - name: Upload logs (if any)
98
+ if: always()
99
+ uses: actions/upload-artifact@v4
100
+ with:
101
+ name: test-logs
102
+ path: |
103
+ test-output.log
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)