@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
@@ -1,620 +0,0 @@
1
- /* Copyright(C) 2021-2024, donavanbecker (https://github.com/donavanbecker). All rights reserved.
2
- *
3
- * lock.ts: @switchbot/homebridge-switchbot.
4
- */
5
- import type { CharacteristicValue, PlatformAccessory, Service } from 'homebridge'
6
- import type { bodyChange, device, lockProServiceData, lockProStatus, lockProWebhookContext, lockServiceData, lockStatus, lockWebhookContext, SwitchBotBLE, SwitchbotDevice, WoSmartLock } from 'node-switchbot'
7
-
8
- import type { SwitchBotHAPPlatform } from '../platform-hap.js'
9
- import type { devicesConfig, lockConfig } from '../settings.js'
10
-
11
- /*
12
- * For Testing Locally:
13
- * import { SwitchBotBLEModel, SwitchBotBLEModelName } from '/Users/Shared/GitHub/OpenWonderLabs/node-switchbot/dist/index.js';
14
- */
15
- import { SwitchBotBLEModel, SwitchBotBLEModelName } from 'node-switchbot'
16
- import { debounceTime, interval, skipWhile, Subject, take, tap } from 'rxjs'
17
-
18
- import { formatDeviceIdAsMac } from '../utils.js'
19
- import { deviceBase } from './device.js'
20
-
21
- export class Lock extends deviceBase {
22
- // Services
23
- private LockMechanism: {
24
- Name: CharacteristicValue
25
- Service: Service
26
- LockTargetState: CharacteristicValue
27
- LockCurrentState: CharacteristicValue
28
- }
29
-
30
- private Battery: {
31
- Name: CharacteristicValue
32
- Service: Service
33
- BatteryLevel: CharacteristicValue
34
- StatusLowBattery: CharacteristicValue
35
- }
36
-
37
- private ContactSensor?: {
38
- Name: CharacteristicValue
39
- Service: Service
40
- ContactSensorState: CharacteristicValue
41
- }
42
-
43
- private Switch?: {
44
- Name: CharacteristicValue
45
- Service: Service
46
- On: CharacteristicValue
47
- }
48
-
49
- // OpenAPI
50
- deviceStatus!: lockStatus | lockProStatus
51
-
52
- // Webhook
53
- webhookContext!: lockWebhookContext | lockProWebhookContext
54
-
55
- // BLE
56
- serviceData!: lockServiceData | lockProServiceData
57
-
58
- // Updates
59
- lockUpdateInProgress!: boolean
60
- doLockUpdate!: Subject<void>
61
-
62
- constructor(
63
- readonly platform: SwitchBotHAPPlatform,
64
- accessory: PlatformAccessory,
65
- device: device & devicesConfig,
66
- ) {
67
- super(platform, accessory, device)
68
- // Set category
69
- accessory.category = this.hap.Categories.DOOR_LOCK
70
-
71
- // this is subject we use to track when we need to POST changes to the SwitchBot API
72
- this.doLockUpdate = new Subject()
73
- this.lockUpdateInProgress = false
74
-
75
- // Initialize LockMechanism Service
76
- accessory.context.LockMechanism = accessory.context.LockMechanism ?? {}
77
- this.LockMechanism = {
78
- Name: accessory.displayName,
79
- Service: accessory.getService(this.hap.Service.LockMechanism) ?? accessory.addService(this.hap.Service.LockMechanism) as Service,
80
- LockTargetState: accessory.context.LockTargetState ?? this.hap.Characteristic.LockTargetState.SECURED,
81
- LockCurrentState: accessory.context.LockCurrentState ?? this.hap.Characteristic.LockCurrentState.SECURED,
82
- }
83
- accessory.context.LockMechanism = this.LockMechanism as object
84
-
85
- // Initialize LockMechanism Characteristics
86
- this.LockMechanism.Service.setCharacteristic(this.hap.Characteristic.Name, this.LockMechanism.Name).getCharacteristic(this.hap.Characteristic.LockTargetState).onGet(() => {
87
- return this.LockMechanism.LockTargetState
88
- }).onSet(this.LockTargetStateSet.bind(this))
89
-
90
- // Initialize Battery property
91
- accessory.context.Battery = accessory.context.Battery ?? {}
92
- this.Battery = {
93
- Name: `${accessory.displayName} Battery`,
94
- Service: accessory.getService(this.hap.Service.Battery) ?? accessory.addService(this.hap.Service.Battery) as Service,
95
- BatteryLevel: accessory.context.BatteryLevel ?? 100,
96
- StatusLowBattery: accessory.context.StatusLowBattery ?? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL,
97
- }
98
- accessory.context.Battery = this.Battery as object
99
-
100
- // Initialize Battery Characteristics
101
- this.Battery.Service.setCharacteristic(this.hap.Characteristic.Name, this.Battery.Name).setCharacteristic(this.hap.Characteristic.ChargingState, this.hap.Characteristic.ChargingState.NOT_CHARGEABLE).getCharacteristic(this.hap.Characteristic.BatteryLevel).onGet(() => {
102
- return this.Battery.BatteryLevel
103
- })
104
-
105
- this.Battery.Service.getCharacteristic(this.hap.Characteristic.StatusLowBattery).onGet(() => {
106
- return this.Battery.StatusLowBattery
107
- })
108
-
109
- // Contact Sensor Service
110
- if ((device as lockConfig).hide_contactsensor) {
111
- if (this.ContactSensor) {
112
- this.debugLog('Removing Contact Sensor Service')
113
- this.ContactSensor.Service = this.accessory.getService(this.hap.Service.ContactSensor) as Service
114
- accessory.removeService(this.ContactSensor.Service)
115
- }
116
- } else {
117
- accessory.context.ContactSensor = accessory.context.ContactSensor ?? {}
118
- this.ContactSensor = {
119
- Name: `${accessory.displayName} Contact Sensor`,
120
- Service: accessory.getService(this.hap.Service.ContactSensor) ?? this.accessory.addService(this.hap.Service.ContactSensor) as Service,
121
- ContactSensorState: accessory.context.ContactSensorState ?? this.hap.Characteristic.ContactSensorState.CONTACT_DETECTED,
122
- }
123
- accessory.context.ContactSensor = this.ContactSensor as object
124
-
125
- // Initialize Contact Sensor Characteristics
126
- this.ContactSensor.Service.setCharacteristic(this.hap.Characteristic.Name, this.ContactSensor.Name).setCharacteristic(this.hap.Characteristic.StatusActive, true).getCharacteristic(this.hap.Characteristic.ContactSensorState).onGet(() => {
127
- return this.ContactSensor!.ContactSensorState
128
- })
129
- }
130
-
131
- // Initialize Latch Button Service
132
- if ((device as lockConfig).activate_latchbutton === false) {
133
- if (this.Switch) {
134
- this.debugLog('Removing Latch Button Service')
135
- this.Switch.Service = accessory.getService(this.hap.Service.Switch) as Service
136
- accessory.removeService(this.Switch.Service)
137
- }
138
- } else {
139
- accessory.context.Switch = accessory.context.Switch ?? {}
140
- this.Switch = {
141
- Name: `${accessory.displayName} Latch`,
142
- Service: accessory.getService(this.hap.Service.Switch) ?? accessory.addService(this.hap.Service.Switch) as Service,
143
- On: accessory.context.On ?? false,
144
- }
145
- accessory.context.Switch = this.Switch as object
146
-
147
- // Initialize Latch Button Characteristics
148
- this.Switch.Service.setCharacteristic(this.hap.Characteristic.Name, this.Switch.Name).getCharacteristic(this.hap.Characteristic.On).onGet(() => {
149
- return this.Switch!.On
150
- }).onSet(this.OnSet.bind(this))
151
- }
152
-
153
- // Retrieve initial values and updateHomekit
154
- try {
155
- this.debugLog('Retrieve initial values and update Homekit')
156
- this.refreshStatus()
157
- } catch (e: any) {
158
- this.errorLog(`failed to retrieve initial values and update Homekit, Error: ${e.message ?? e}`)
159
- }
160
-
161
- // regisiter webhook event handler if enabled
162
- try {
163
- this.debugLog('Registering Webhook Event Handler')
164
- this.registerWebhook()
165
- } catch (e: any) {
166
- this.errorLog(`failed to registerWebhook, Error: ${e.message ?? e}`)
167
- }
168
-
169
- // regisiter platform BLE event handler if enabled
170
- try {
171
- this.debugLog('Registering Platform BLE Event Handler')
172
- this.registerPlatformBLE()
173
- } catch (e: any) {
174
- this.errorLog(`failed to registerPlatformBLE, Error: ${e.message ?? e}`)
175
- }
176
-
177
- // Start an update interval
178
- interval(this.deviceRefreshRate * 1000)
179
- .pipe(skipWhile(() => this.lockUpdateInProgress))
180
- .subscribe(async () => {
181
- await this.refreshStatus()
182
- })
183
-
184
- // Watch for Lock change events
185
- // We put in a debounce of 100ms so we don't make duplicate calls
186
- this.doLockUpdate
187
- .pipe(
188
- tap(() => {
189
- this.lockUpdateInProgress = true
190
- }),
191
- debounceTime(this.devicePushRate * 1000),
192
- )
193
- .subscribe(async () => {
194
- try {
195
- await this.pushChanges()
196
- } catch (e: any) {
197
- await this.apiError(e)
198
- this.errorLog(`failed pushChanges with ${device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
199
- }
200
- this.lockUpdateInProgress = false
201
- })
202
- }
203
-
204
- async BLEparseStatus(): Promise<void> {
205
- this.debugLog('BLEparseStatus')
206
- this.debugLog(`(lockState) = BLE:(${this.serviceData.status}), current:(${this.LockMechanism.LockCurrentState})`)
207
-
208
- // LockCurrentState
209
- this.LockMechanism.LockCurrentState = this.serviceData.status === 'locked'
210
- ? this.hap.Characteristic.LockCurrentState.SECURED
211
- : this.hap.Characteristic.LockCurrentState.UNSECURED
212
- this.debugLog(`LockCurrentState: ${this.LockMechanism.LockCurrentState}`)
213
-
214
- // LockTargetState
215
- this.LockMechanism.LockTargetState = this.serviceData.status === 'locked'
216
- ? this.hap.Characteristic.LockTargetState.SECURED
217
- : this.hap.Characteristic.LockTargetState.UNSECURED
218
- this.debugLog(`LockTargetState: ${this.LockMechanism.LockTargetState}`)
219
-
220
- // Contact Sensor
221
- if (!(this.device as lockConfig).hide_contactsensor && this.ContactSensor?.Service) {
222
- this.ContactSensor.ContactSensorState = this.serviceData.door_open
223
- ? this.hap.Characteristic.ContactSensorState.CONTACT_NOT_DETECTED
224
- : this.hap.Characteristic.ContactSensorState.CONTACT_DETECTED
225
- this.debugLog(`ContactSensorState: ${this.ContactSensor.ContactSensorState}`)
226
- }
227
- // Battery Info
228
- if ('battery' in this.serviceData) {
229
- // BatteryLevel
230
- this.Battery.BatteryLevel = this.serviceData.battery
231
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
232
- // StatusLowBattery
233
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
234
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
235
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
236
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
237
- }
238
- }
239
-
240
- async openAPIparseStatus(): Promise<void> {
241
- this.debugLog('openAPIparseStatus')
242
- this.debugLog(`(lockState) = OpenAPI:(${this.deviceStatus.lockState}), current:(${this.LockMechanism.LockCurrentState})`)
243
-
244
- // LockCurrentState
245
- this.LockMechanism.LockCurrentState = this.deviceStatus.lockState === 'locked'
246
- ? this.hap.Characteristic.LockCurrentState.SECURED
247
- : this.hap.Characteristic.LockCurrentState.UNSECURED
248
- this.debugLog(`LockCurrentState: ${this.LockMechanism.LockCurrentState}`)
249
-
250
- // LockTargetState
251
- this.LockMechanism.LockTargetState = this.deviceStatus.lockState === 'locked'
252
- ? this.hap.Characteristic.LockTargetState.SECURED
253
- : this.hap.Characteristic.LockTargetState.UNSECURED
254
- this.debugLog(`LockTargetState: ${this.LockMechanism.LockTargetState}`)
255
-
256
- // ContactSensorState
257
- if (!(this.device as lockConfig).hide_contactsensor && this.ContactSensor?.Service) {
258
- this.ContactSensor.ContactSensorState = this.deviceStatus.doorState === 'opened'
259
- ? this.hap.Characteristic.ContactSensorState.CONTACT_NOT_DETECTED
260
- : this.hap.Characteristic.ContactSensorState.CONTACT_DETECTED
261
- this.debugLog(`ContactSensorState: ${this.ContactSensor.ContactSensorState}`)
262
- }
263
-
264
- // BatteryLevel
265
- this.Battery.BatteryLevel = this.deviceStatus.battery
266
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
267
-
268
- // StatusLowBattery
269
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
270
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
271
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
272
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
273
-
274
- // Firmware Version
275
- if (this.deviceStatus.version) {
276
- const version = this.deviceStatus.version.toString()
277
- this.debugLog(`Firmware Version: ${version.replace(/^V|-.*$/g, '')}`)
278
- const deviceVersion = version.replace(/^V|-.*$/g, '') ?? '0.0.0'
279
- this.accessory
280
- .getService(this.hap.Service.AccessoryInformation)!
281
- .setCharacteristic(this.hap.Characteristic.HardwareRevision, deviceVersion)
282
- .setCharacteristic(this.hap.Characteristic.FirmwareRevision, deviceVersion)
283
- .getCharacteristic(this.hap.Characteristic.FirmwareRevision)
284
- .updateValue(deviceVersion)
285
- this.accessory.context.version = deviceVersion
286
- this.debugSuccessLog(`version: ${this.accessory.context.version}`)
287
- }
288
- }
289
-
290
- async parseStatusWebhook(): Promise<void> {
291
- this.debugLog('parseStatusWebhook')
292
- this.debugLog(`(lockState) = Webhook:(${this.webhookContext.lockState}), current:(${this.LockMechanism.LockCurrentState})`)
293
-
294
- // LockCurrentState
295
- this.LockMechanism.LockCurrentState = this.webhookContext.lockState === 'LOCKED'
296
- ? this.hap.Characteristic.LockCurrentState.SECURED
297
- : this.hap.Characteristic.LockCurrentState.UNSECURED
298
- this.debugLog(`LockCurrentState: ${this.LockMechanism.LockCurrentState}`)
299
-
300
- // LockTargetState
301
- this.LockMechanism.LockTargetState = this.webhookContext.lockState === 'LOCKED'
302
- ? this.hap.Characteristic.LockTargetState.SECURED
303
- : this.hap.Characteristic.LockTargetState.UNSECURED
304
- this.debugLog(`LockTargetState: ${this.LockMechanism.LockTargetState}`)
305
- }
306
-
307
- /**
308
- * Asks the SwitchBot API for the latest device information
309
- */
310
- async refreshStatus(): Promise<void> {
311
- if (this.BLE) {
312
- await this.BLERefreshStatus()
313
- } else if (this.OpenAPI && this.platform.config.credentials?.token) {
314
- await this.openAPIRefreshStatus()
315
- } else {
316
- await this.offlineOff()
317
- this.debugWarnLog(`Connection Type: ${this.device.connectionType}, refreshStatus will not happen.`)
318
- }
319
- }
320
-
321
- async BLERefreshStatus(): Promise<void> {
322
- this.debugLog('BLERefreshStatus')
323
- const switchBotBLE = await this.switchbotBLE()
324
- if (switchBotBLE === undefined) {
325
- await this.BLERefreshConnection(switchBotBLE)
326
- } else {
327
- // Start to monitor advertisement packets
328
- (async () => {
329
- // Start to monitor advertisement packets
330
- const serviceData = await this.monitorAdvertisementPackets(switchBotBLE) as lockServiceData | lockProServiceData
331
- // Update HomeKit
332
- if ((serviceData.model === SwitchBotBLEModel.Lock || SwitchBotBLEModel.LockPro)
333
- && (serviceData.modelName === SwitchBotBLEModelName.Lock || SwitchBotBLEModelName.LockPro)) {
334
- this.serviceData = serviceData
335
- if (serviceData !== undefined || serviceData !== null) {
336
- await this.BLEparseStatus()
337
- await this.updateHomeKitCharacteristics()
338
- } else {
339
- this.errorLog(`serviceData is either undefined or null, serviceData: ${JSON.stringify(serviceData)}`)
340
- await this.BLERefreshConnection(switchBotBLE)
341
- }
342
- } else {
343
- this.errorLog(`failed to get serviceData, serviceData: ${JSON.stringify(serviceData)}`)
344
- await this.BLERefreshConnection(switchBotBLE)
345
- }
346
- })()
347
- }
348
- }
349
-
350
- async registerPlatformBLE(): Promise<void> {
351
- this.debugLog('registerPlatformBLE')
352
- if (this.config.options?.BLE && !this.device.disablePlatformBLE) {
353
- this.debugLog('is listening to Platform BLE.')
354
- try {
355
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
356
- this.device.bleMac = formattedDeviceId
357
- this.debugLog(`bleMac: ${this.device.bleMac}`)
358
- this.platform.bleEventHandler[this.device.bleMac] = async (context: lockServiceData | lockProServiceData) => {
359
- try {
360
- this.serviceData = context
361
- if (context !== undefined || context !== null) {
362
- this.debugLog(`received BLE: ${JSON.stringify(context)}`)
363
- await this.BLEparseStatus()
364
- await this.updateHomeKitCharacteristics()
365
- } else {
366
- this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`)
367
- await this.BLERefreshConnection(context)
368
- }
369
- } catch (e: any) {
370
- this.errorLog(`failed to handle BLE. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`)
371
- }
372
- }
373
- } catch (error) {
374
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
375
- }
376
- } else {
377
- this.debugLog('is not listening to Platform BLE')
378
- }
379
- }
380
-
381
- async openAPIRefreshStatus(): Promise<void> {
382
- this.debugLog('openAPIRefreshStatus')
383
- try {
384
- const deviceStatus = await this.deviceRefreshStatus<lockStatus | lockProStatus>()
385
- this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
386
- if (await this.successfulStatusCodes(deviceStatus)) {
387
- this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
388
- this.deviceStatus = deviceStatus.body
389
- await this.openAPIparseStatus()
390
- await this.updateHomeKitCharacteristics()
391
- } else {
392
- this.debugWarnLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
393
- }
394
- } catch (e: any) {
395
- await this.apiError(e)
396
- this.errorLog(`failed openAPIRefreshStatus with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
397
- }
398
- }
399
-
400
- async registerWebhook() {
401
- if (this.device.webhook) {
402
- this.debugLog('is listening webhook.')
403
- this.platform.webhookEventHandler[this.device.deviceId] = async (context: lockWebhookContext | lockProWebhookContext) => {
404
- try {
405
- this.webhookContext = context
406
- if (context !== undefined || context !== null) {
407
- this.debugLog(`received Webhook: ${JSON.stringify(context)}`)
408
- await this.parseStatusWebhook()
409
- await this.updateHomeKitCharacteristics()
410
- } else {
411
- this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`)
412
- }
413
- } catch (e: any) {
414
- this.errorLog(`failed to handle webhook. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`)
415
- }
416
- }
417
- } else {
418
- this.debugLog('is not listening webhook.')
419
- }
420
- }
421
-
422
- /**
423
- * Pushes the requested changes to the SwitchBot API
424
- * deviceType commandType Command command parameter Description
425
- * Lock - "command" "lock" "default" = set to ???? state
426
- * Lock - "command" "unlock" "default" = set to ???? state - LockCurrentState
427
- */
428
- async pushChanges(): Promise<void> {
429
- if (this.BLE) {
430
- await this.BLEpushChanges()
431
- } else if (this.OpenAPI && this.platform.config.credentials?.token) {
432
- await this.openAPIpushChanges()
433
- } else {
434
- await this.offlineOff()
435
- this.debugWarnLog(`Connection Type: ${this.device.connectionType}, pushChanges will not happen.`)
436
- }
437
- // Refresh the status from the API
438
- interval(15000)
439
- .pipe(skipWhile(() => this.lockUpdateInProgress))
440
- .pipe(take(1))
441
- .subscribe(async () => {
442
- await this.refreshStatus()
443
- })
444
- }
445
-
446
- async BLEpushChanges(): Promise<void> {
447
- this.debugLog('BLEpushChanges')
448
- if (this.LockMechanism.LockTargetState !== this.accessory.context.LockTargetState) {
449
- const switchBotBLE = await this.platform.connectBLE(this.accessory, this.device)
450
- try {
451
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
452
- this.device.bleMac = formattedDeviceId
453
- this.debugLog(`bleMac: ${this.device.bleMac}`)
454
- if (switchBotBLE !== false) {
455
- switchBotBLE
456
- .discover({ model: this.device.bleModel, id: this.device.bleMac })
457
- .then(async (device_list: SwitchbotDevice[]) => {
458
- return await this.retryBLE({
459
- max: this.maxRetryBLE(),
460
- fn: async () => {
461
- if (this.LockMechanism.LockTargetState === this.hap.Characteristic.LockTargetState.SECURED) {
462
- return await (device_list[0] as WoSmartLock).lock()
463
- } else {
464
- return await (device_list[0] as WoSmartLock).unlock()
465
- }
466
- },
467
- })
468
- })
469
- .then(async () => {
470
- this.successLog(`LockTargetState: ${this.LockMechanism.LockTargetState} sent over SwitchBot BLE, sent successfully`)
471
- await this.updateHomeKitCharacteristics()
472
- })
473
- .catch(async (e: any) => {
474
- await this.apiError(e)
475
- this.errorLog(`failed BLEpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
476
- await this.BLEPushConnection()
477
- })
478
- } else {
479
- this.errorLog(`wasn't able to establish BLE Connection, node-switchbot: ${JSON.stringify(switchBotBLE)}`)
480
- await this.BLEPushConnection()
481
- }
482
- } catch (error) {
483
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
484
- }
485
- } else {
486
- this.debugLog(`No changes (BLEpushChanges), LockTargetState: ${this.LockMechanism.LockTargetState}, LockCurrentState: ${this.LockMechanism.LockCurrentState}`)
487
- }
488
- }
489
-
490
- async openAPIpushChanges(LatchUnlock?: boolean): Promise<void> {
491
- this.debugLog('openAPIpushChanges')
492
- if ((this.LockMechanism.LockTargetState !== this.accessory.context.LockTargetState) || LatchUnlock) {
493
- // Determine the command based on the LockTargetState or the forceUnlock parameter
494
- const command = LatchUnlock ? 'unlock' : this.LockMechanism.LockTargetState ? 'lock' : 'unlock'
495
- const bodyChange: bodyChange = {
496
- command: `${command}`,
497
- parameter: 'default',
498
- commandType: 'command',
499
- }
500
- this.debugLog(`SwitchBot OpenAPI bodyChange: ${JSON.stringify(bodyChange)}`)
501
- try {
502
- const deviceStatus = await this.pushChangeRequest(bodyChange)
503
- this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
504
- if (await this.successfulStatusCodes(deviceStatus)) {
505
- this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
506
- await this.updateHomeKitCharacteristics()
507
- } else {
508
- await this.statusCode(deviceStatus.statusCode)
509
- }
510
- } catch (e: any) {
511
- await this.apiError(e)
512
- this.errorLog(`failed openAPIpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
513
- }
514
- } else {
515
- this.debugLog(`No changes (openAPIpushChanges), LockCurrentState: ${this.LockMechanism.LockCurrentState}, TargetPosition: ${this.LockMechanism.LockTargetState}`)
516
- }
517
- }
518
-
519
- /**
520
- * Handle requests to set the value of the "On" characteristic
521
- */
522
- async LockTargetStateSet(value: CharacteristicValue): Promise<void> {
523
- if (this.LockMechanism.LockTargetState !== this.accessory.context.LockTargetState) {
524
- this.infoLog(`Set LockTargetState: ${value}`)
525
- } else {
526
- this.debugLog(`No Changes, LockTargetState: ${value}`)
527
- }
528
-
529
- this.LockMechanism.LockTargetState = value
530
- this.doLockUpdate.next()
531
- }
532
-
533
- /**
534
- * Handle requests to set the value of the "On" characteristic
535
- */
536
- async OnSet(value: CharacteristicValue): Promise<void> {
537
- this.debugLog(`Latch Button Set On: ${value}`)
538
- if (value) {
539
- this.debugLog('Attempting to open the latch')
540
-
541
- this.openAPIpushChanges(value as boolean).then(async () => {
542
- this.debugLog('Latch opened successfully')
543
- this.debugLog(`SwitchService is: ${this.Switch?.Service ? 'available' : 'not available'}`)
544
-
545
- // simulate button press to turn the switch back off
546
- if (this.Switch?.Service) {
547
- const SwitchService = this.Switch.Service
548
- // Simulate a button press by waiting a short period before turning the switch off
549
- setTimeout(async () => {
550
- SwitchService.getCharacteristic(this.hap.Characteristic.On).updateValue(false)
551
- this.debugLog('Latch button switched off automatically.')
552
- }, 500) // 500 ms delay
553
- }
554
- }).catch(async (e: any) => {
555
- // Log the error if the operation failed
556
- this.debugLog(`Error opening latch: ${e.message ?? e}`)
557
- // Ensure we turn the switch back off even in case of an error
558
- if (this.Switch?.Service) {
559
- this.Switch.Service.getCharacteristic(this.hap.Characteristic.On).updateValue(false)
560
- this.debugLog('Latch button switched off after an error.')
561
- }
562
- })
563
- } else {
564
- this.debugLog('Switch is off, nothing to do')
565
- }
566
-
567
- this.Switch!.On = value
568
- this.doLockUpdate.next()
569
- }
570
-
571
- async updateHomeKitCharacteristics(): Promise<void> {
572
- // LockCurrentState
573
- await this.updateCharacteristic(this.LockMechanism.Service, this.hap.Characteristic.LockTargetState, this.LockMechanism.LockTargetState, 'LockTargetState')
574
- // LockCurrentState
575
- await this.updateCharacteristic(this.LockMechanism.Service, this.hap.Characteristic.LockCurrentState, this.LockMechanism.LockCurrentState, 'LockCurrentState')
576
- // ContactSensorState
577
- if (!(this.device as lockConfig).hide_contactsensor && this.ContactSensor?.Service) {
578
- await this.updateCharacteristic(this.ContactSensor.Service, this.hap.Characteristic.ContactSensorState, this.ContactSensor.ContactSensorState, 'ContactSensorState')
579
- }
580
- // BatteryLevel
581
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.BatteryLevel, this.Battery.BatteryLevel, 'BatteryLevel')
582
- // StatusLowBattery
583
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.StatusLowBattery, this.Battery.StatusLowBattery, 'StatusLowBattery')
584
- }
585
-
586
- async BLEPushConnection() {
587
- if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
588
- this.warnLog('Using OpenAPI Connection to Push Changes')
589
- await this.openAPIpushChanges()
590
- }
591
- }
592
-
593
- async BLERefreshConnection(switchbot: SwitchBotBLE): Promise<void> {
594
- this.errorLog(`wasn't able to establish BLE Connection, node-switchbot: ${switchbot}`)
595
- if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
596
- this.warnLog('Using OpenAPI Connection to Refresh Status')
597
- await this.openAPIRefreshStatus()
598
- }
599
- }
600
-
601
- async offlineOff(): Promise<void> {
602
- if (this.device.offline) {
603
- this.LockMechanism.Service.updateCharacteristic(this.hap.Characteristic.LockTargetState, this.hap.Characteristic.LockTargetState.SECURED)
604
- this.LockMechanism.Service.updateCharacteristic(this.hap.Characteristic.LockCurrentState, this.hap.Characteristic.LockCurrentState.SECURED)
605
- if (!(this.device as lockConfig).hide_contactsensor && this.ContactSensor?.Service) {
606
- this.ContactSensor.Service.updateCharacteristic(this.hap.Characteristic.ContactSensorState, this.hap.Characteristic.ContactSensorState.CONTACT_DETECTED)
607
- }
608
- }
609
- }
610
-
611
- async apiError(e: any): Promise<void> {
612
- this.LockMechanism.Service.updateCharacteristic(this.hap.Characteristic.LockTargetState, e)
613
- this.LockMechanism.Service.updateCharacteristic(this.hap.Characteristic.LockCurrentState, e)
614
- if (!(this.device as lockConfig).hide_contactsensor && this.ContactSensor?.Service) {
615
- this.ContactSensor.Service.updateCharacteristic(this.hap.Characteristic.ContactSensorState, e)
616
- }
617
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.BatteryLevel, e)
618
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.StatusLowBattery, e)
619
- }
620
- }