@switchbot/homebridge-switchbot 5.0.0-beta.9 → 5.0.0-beta.91
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.
- package/.github/ISSUE_TEMPLATE/e2e-verification.md +36 -0
- package/.github/workflows/ci.yml +32 -0
- package/.github/workflows/manual-e2e.yml +115 -0
- package/.github/workflows/release.yml +0 -4
- package/CHANGELOG.md +35 -0
- package/E2E-VERIFICATION.md +121 -0
- package/MIGRATION.md +44 -0
- package/README.md +56 -3
- package/config.schema.json +91 -14787
- package/dist/deviceFactory.d.ts +13 -0
- package/dist/deviceFactory.d.ts.map +1 -0
- package/dist/deviceFactory.js +81 -0
- package/dist/deviceFactory.js.map +1 -0
- package/dist/devices/deviceBase.d.ts +50 -0
- package/dist/devices/deviceBase.d.ts.map +1 -0
- package/dist/devices/deviceBase.js +119 -0
- package/dist/devices/deviceBase.js.map +1 -0
- package/dist/devices/genericDevice.d.ts +283 -0
- package/dist/devices/genericDevice.d.ts.map +1 -0
- package/dist/devices/genericDevice.js +1035 -0
- package/dist/devices/genericDevice.js.map +1 -0
- package/dist/homebridge-ui/public/index.html +630 -246
- package/dist/homebridge-ui/server.d.ts +3 -1
- package/dist/homebridge-ui/server.d.ts.map +1 -1
- package/dist/homebridge-ui/server.js +367 -36
- package/dist/homebridge-ui/server.js.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4 -32
- package/dist/index.js.map +1 -1
- package/dist/platform.d.ts +35 -0
- package/dist/platform.d.ts.map +1 -0
- package/dist/platform.js +514 -0
- package/dist/platform.js.map +1 -0
- package/dist/settings.d.ts +10 -249
- package/dist/settings.d.ts.map +1 -1
- package/dist/settings.js +5 -30
- package/dist/settings.js.map +1 -1
- package/dist/switchbotClient.d.ts +32 -0
- package/dist/switchbotClient.d.ts.map +1 -0
- package/dist/switchbotClient.js +194 -0
- package/dist/switchbotClient.js.map +1 -0
- package/dist/utils.d.ts +39 -50
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +39 -688
- package/dist/utils.js.map +1 -1
- package/docs/assets/highlight.css +14 -0
- package/docs/assets/icons.js +1 -1
- package/docs/assets/icons.svg +1 -1
- package/docs/assets/main.js +2 -2
- package/docs/assets/style.css +3 -3
- package/docs/index.html +77 -13
- package/docs/variables/default.html +1 -1
- package/eslint.config.js +2 -8
- package/package.json +21 -28
- package/scripts/e2e/README.md +25 -0
- package/scripts/e2e/curtain-e2e.sh +70 -0
- package/scripts/e2e/fan-e2e.sh +75 -0
- package/scripts/e2e/light-advanced-e2e.sh +97 -0
- package/scripts/e2e/light-e2e.sh +75 -0
- package/scripts/e2e/list-accessories.sh +19 -0
- package/scripts/e2e/lock-e2e.sh +65 -0
- package/scripts/generate-matter-maps.js +60 -0
- package/scripts/run-e2e-local.sh +14 -0
- package/src/deviceFactory.ts +122 -0
- package/src/devices/deviceBase.ts +141 -0
- package/src/devices/genericDevice.ts +965 -0
- package/src/homebridge-ui/public/index.html +630 -246
- package/src/homebridge-ui/server.ts +434 -41
- package/src/index.ts +4 -33
- package/src/platform.ts +515 -0
- package/src/settings.ts +12 -277
- package/src/switchbotClient.ts +203 -0
- package/src/utils.ts +45 -713
- package/test/accessory-restore.spec.ts +73 -0
- package/test/device-mapping.spec.ts +37 -0
- package/test/deviceFactory.spec.ts +18 -0
- package/test/e2e/run-e2e.spec.ts +50 -0
- package/test/fan-swing.spec.ts +29 -0
- package/test/helpers/matter-harness.ts +53 -0
- package/test/lock-users.spec.ts +44 -0
- package/test/matter-childbridge.spec.ts +55 -0
- package/test/matter-descriptors.spec.ts +97 -0
- package/test/matter-device-state.spec.ts +101 -0
- package/test/matter-integration.spec.ts +70 -0
- package/test/platform.integration.spec.ts +55 -0
- package/test/switchbot-client-debounce.spec.ts +131 -0
- package/test/switchbot-client-openapi.spec.ts +56 -0
- package/test/switchbotClient.spec.ts +10 -0
- package/test/utils.spec.ts +20 -0
- package/vitest.config.ts +7 -0
- package/coverage/base.css +0 -224
- package/coverage/block-navigation.js +0 -87
- package/coverage/clover.xml +0 -15847
- package/coverage/coverage-final.json +0 -42
- package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
- package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
- package/coverage/docs/assets/dmt/index.html +0 -131
- package/coverage/docs/assets/hierarchy.js.html +0 -85
- package/coverage/docs/assets/icons.js.html +0 -136
- package/coverage/docs/assets/index.html +0 -146
- package/coverage/docs/assets/main.js.html +0 -265
- package/coverage/favicon.png +0 -0
- package/coverage/index.html +0 -191
- package/coverage/prettify.css +0 -1
- package/coverage/prettify.js +0 -2
- package/coverage/sort-arrow-sprite.png +0 -0
- package/coverage/sorter.js +0 -196
- package/coverage/src/device/blindtilt.ts.html +0 -3238
- package/coverage/src/device/bot.ts.html +0 -2803
- package/coverage/src/device/ceilinglight.ts.html +0 -2338
- package/coverage/src/device/colorbulb.ts.html +0 -2824
- package/coverage/src/device/contact.ts.html +0 -1465
- package/coverage/src/device/curtain.ts.html +0 -2869
- package/coverage/src/device/device.ts.html +0 -2500
- package/coverage/src/device/fan.ts.html +0 -2242
- package/coverage/src/device/hub.ts.html +0 -1408
- package/coverage/src/device/humidifier.ts.html +0 -2116
- package/coverage/src/device/index.html +0 -416
- package/coverage/src/device/iosensor.ts.html +0 -1375
- package/coverage/src/device/lightstrip.ts.html +0 -2617
- package/coverage/src/device/lock.ts.html +0 -1963
- package/coverage/src/device/meter.ts.html +0 -1372
- package/coverage/src/device/meterplus.ts.html +0 -1384
- package/coverage/src/device/meterpro.ts.html +0 -1618
- package/coverage/src/device/motion.ts.html +0 -1264
- package/coverage/src/device/plug.ts.html +0 -1372
- package/coverage/src/device/relayswitch.ts.html +0 -2284
- package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
- package/coverage/src/device/waterdetector.ts.html +0 -1294
- package/coverage/src/homebridge-ui/index.html +0 -116
- package/coverage/src/homebridge-ui/server.ts.html +0 -229
- package/coverage/src/index.html +0 -161
- package/coverage/src/index.ts.html +0 -124
- package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
- package/coverage/src/irdevice/airpurifier.ts.html +0 -844
- package/coverage/src/irdevice/camera.ts.html +0 -475
- package/coverage/src/irdevice/fan.ts.html +0 -766
- package/coverage/src/irdevice/index.html +0 -251
- package/coverage/src/irdevice/irdevice.ts.html +0 -1117
- package/coverage/src/irdevice/light.ts.html +0 -826
- package/coverage/src/irdevice/other.ts.html +0 -2458
- package/coverage/src/irdevice/tv.ts.html +0 -1222
- package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
- package/coverage/src/irdevice/waterheater.ts.html +0 -469
- package/coverage/src/platform.ts.html +0 -8776
- package/coverage/src/settings.ts.html +0 -934
- package/coverage/src/utils.ts.html +0 -2092
- package/dist/devices-hap/airpurifier.d.ts +0 -54
- package/dist/devices-hap/airpurifier.d.ts.map +0 -1
- package/dist/devices-hap/airpurifier.js +0 -527
- package/dist/devices-hap/airpurifier.js.map +0 -1
- package/dist/devices-hap/blindtilt.d.ts +0 -90
- package/dist/devices-hap/blindtilt.d.ts.map +0 -1
- package/dist/devices-hap/blindtilt.js +0 -974
- package/dist/devices-hap/blindtilt.js.map +0 -1
- package/dist/devices-hap/bot.d.ts +0 -102
- package/dist/devices-hap/bot.d.ts.map +0 -1
- package/dist/devices-hap/bot.js +0 -811
- package/dist/devices-hap/bot.js.map +0 -1
- package/dist/devices-hap/ceilinglight.d.ts +0 -85
- package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
- package/dist/devices-hap/ceilinglight.js +0 -701
- package/dist/devices-hap/ceilinglight.js.map +0 -1
- package/dist/devices-hap/colorbulb.d.ts +0 -88
- package/dist/devices-hap/colorbulb.d.ts.map +0 -1
- package/dist/devices-hap/colorbulb.js +0 -881
- package/dist/devices-hap/colorbulb.js.map +0 -1
- package/dist/devices-hap/contact.d.ts +0 -44
- package/dist/devices-hap/contact.d.ts.map +0 -1
- package/dist/devices-hap/contact.js +0 -409
- package/dist/devices-hap/contact.js.map +0 -1
- package/dist/devices-hap/curtain.d.ts +0 -73
- package/dist/devices-hap/curtain.d.ts.map +0 -1
- package/dist/devices-hap/curtain.js +0 -869
- package/dist/devices-hap/curtain.js.map +0 -1
- package/dist/devices-hap/device.d.ts +0 -98
- package/dist/devices-hap/device.d.ts.map +0 -1
- package/dist/devices-hap/device.js +0 -749
- package/dist/devices-hap/device.js.map +0 -1
- package/dist/devices-hap/fan.d.ts +0 -69
- package/dist/devices-hap/fan.d.ts.map +0 -1
- package/dist/devices-hap/fan.js +0 -649
- package/dist/devices-hap/fan.js.map +0 -1
- package/dist/devices-hap/hub.d.ts +0 -37
- package/dist/devices-hap/hub.d.ts.map +0 -1
- package/dist/devices-hap/hub.js +0 -392
- package/dist/devices-hap/hub.js.map +0 -1
- package/dist/devices-hap/humidifier.d.ts +0 -68
- package/dist/devices-hap/humidifier.d.ts.map +0 -1
- package/dist/devices-hap/humidifier.js +0 -628
- package/dist/devices-hap/humidifier.js.map +0 -1
- package/dist/devices-hap/iosensor.d.ts +0 -42
- package/dist/devices-hap/iosensor.d.ts.map +0 -1
- package/dist/devices-hap/iosensor.js +0 -382
- package/dist/devices-hap/iosensor.js.map +0 -1
- package/dist/devices-hap/lightstrip.d.ts +0 -79
- package/dist/devices-hap/lightstrip.d.ts.map +0 -1
- package/dist/devices-hap/lightstrip.js +0 -797
- package/dist/devices-hap/lightstrip.js.map +0 -1
- package/dist/devices-hap/lock.d.ts +0 -53
- package/dist/devices-hap/lock.d.ts.map +0 -1
- package/dist/devices-hap/lock.js +0 -561
- package/dist/devices-hap/lock.js.map +0 -1
- package/dist/devices-hap/meter.d.ts +0 -37
- package/dist/devices-hap/meter.d.ts.map +0 -1
- package/dist/devices-hap/meter.js +0 -379
- package/dist/devices-hap/meter.js.map +0 -1
- package/dist/devices-hap/meterplus.d.ts +0 -42
- package/dist/devices-hap/meterplus.d.ts.map +0 -1
- package/dist/devices-hap/meterplus.js +0 -384
- package/dist/devices-hap/meterplus.js.map +0 -1
- package/dist/devices-hap/meterpro.d.ts +0 -43
- package/dist/devices-hap/meterpro.d.ts.map +0 -1
- package/dist/devices-hap/meterpro.js +0 -468
- package/dist/devices-hap/meterpro.js.map +0 -1
- package/dist/devices-hap/motion.d.ts +0 -42
- package/dist/devices-hap/motion.d.ts.map +0 -1
- package/dist/devices-hap/motion.js +0 -345
- package/dist/devices-hap/motion.js.map +0 -1
- package/dist/devices-hap/plug.d.ts +0 -49
- package/dist/devices-hap/plug.d.ts.map +0 -1
- package/dist/devices-hap/plug.js +0 -395
- package/dist/devices-hap/plug.js.map +0 -1
- package/dist/devices-hap/relayswitch.d.ts +0 -96
- package/dist/devices-hap/relayswitch.d.ts.map +0 -1
- package/dist/devices-hap/relayswitch.js +0 -642
- package/dist/devices-hap/relayswitch.js.map +0 -1
- package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
- package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
- package/dist/devices-hap/robotvacuumcleaner.js +0 -523
- package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
- package/dist/devices-hap/waterdetector.d.ts +0 -41
- package/dist/devices-hap/waterdetector.d.ts.map +0 -1
- package/dist/devices-hap/waterdetector.js +0 -356
- package/dist/devices-hap/waterdetector.js.map +0 -1
- package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -63
- package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
- package/dist/devices-matter/BaseMatterAccessory.js +0 -100
- package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
- package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
- package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ColorLightAccessory.js +0 -95
- package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
- package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
- package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -78
- package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
- package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ContactSensorAccessory.js +0 -34
- package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
- package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
- package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/DimmableLightAccessory.js +0 -167
- package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
- package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
- package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
- package/dist/devices-matter/DoorLockAccessory.js +0 -50
- package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
- package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
- package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -107
- package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
- package/dist/devices-matter/FanAccessory.d.ts +0 -16
- package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
- package/dist/devices-matter/FanAccessory.js +0 -81
- package/dist/devices-matter/FanAccessory.js.map +0 -1
- package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
- package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
- package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
- package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/LeakSensorAccessory.js +0 -33
- package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
- package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/LightSensorAccessory.js +0 -34
- package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
- package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
- package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
- package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
- package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
- package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OnOffLightAccessory.js +0 -118
- package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
- package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -12
- package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OnOffOutletAccessory.js +0 -40
- package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
- package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
- package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
- package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
- package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -68
- package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
- package/dist/devices-matter/RoboticVacuumAccessory.js +0 -334
- package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
- package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
- package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
- package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
- package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
- package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
- package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
- package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
- package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ThermostatAccessory.js +0 -95
- package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
- package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
- package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
- package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
- package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
- package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
- package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
- package/dist/devices-matter/WindowBlindAccessory.js +0 -80
- package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
- package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
- package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
- package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
- package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
- package/dist/devices-matter/custom/index.d.ts +0 -8
- package/dist/devices-matter/custom/index.d.ts.map +0 -1
- package/dist/devices-matter/custom/index.js +0 -8
- package/dist/devices-matter/custom/index.js.map +0 -1
- package/dist/devices-matter/index.d.ts +0 -29
- package/dist/devices-matter/index.d.ts.map +0 -1
- package/dist/devices-matter/index.js +0 -28
- package/dist/devices-matter/index.js.map +0 -1
- package/dist/index.test.d.ts +0 -2
- package/dist/index.test.d.ts.map +0 -1
- package/dist/index.test.js +0 -14
- package/dist/index.test.js.map +0 -1
- package/dist/irdevice/airconditioner.d.ts +0 -61
- package/dist/irdevice/airconditioner.d.ts.map +0 -1
- package/dist/irdevice/airconditioner.js +0 -472
- package/dist/irdevice/airconditioner.js.map +0 -1
- package/dist/irdevice/airpurifier.d.ts +0 -50
- package/dist/irdevice/airpurifier.d.ts.map +0 -1
- package/dist/irdevice/airpurifier.js +0 -213
- package/dist/irdevice/airpurifier.js.map +0 -1
- package/dist/irdevice/camera.d.ts +0 -32
- package/dist/irdevice/camera.d.ts.map +0 -1
- package/dist/irdevice/camera.js +0 -107
- package/dist/irdevice/camera.js.map +0 -1
- package/dist/irdevice/fan.d.ts +0 -36
- package/dist/irdevice/fan.d.ts.map +0 -1
- package/dist/irdevice/fan.js +0 -200
- package/dist/irdevice/fan.js.map +0 -1
- package/dist/irdevice/irdevice.d.ts +0 -68
- package/dist/irdevice/irdevice.d.ts.map +0 -1
- package/dist/irdevice/irdevice.js +0 -298
- package/dist/irdevice/irdevice.js.map +0 -1
- package/dist/irdevice/light.d.ts +0 -36
- package/dist/irdevice/light.d.ts.map +0 -1
- package/dist/irdevice/light.js +0 -206
- package/dist/irdevice/light.js.map +0 -1
- package/dist/irdevice/other.d.ts +0 -57
- package/dist/irdevice/other.d.ts.map +0 -1
- package/dist/irdevice/other.js +0 -778
- package/dist/irdevice/other.js.map +0 -1
- package/dist/irdevice/tv.d.ts +0 -45
- package/dist/irdevice/tv.d.ts.map +0 -1
- package/dist/irdevice/tv.js +0 -327
- package/dist/irdevice/tv.js.map +0 -1
- package/dist/irdevice/vacuumcleaner.d.ts +0 -28
- package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
- package/dist/irdevice/vacuumcleaner.js +0 -104
- package/dist/irdevice/vacuumcleaner.js.map +0 -1
- package/dist/irdevice/waterheater.d.ts +0 -30
- package/dist/irdevice/waterheater.d.ts.map +0 -1
- package/dist/irdevice/waterheater.js +0 -105
- package/dist/irdevice/waterheater.js.map +0 -1
- package/dist/platform-hap.d.ts +0 -149
- package/dist/platform-hap.d.ts.map +0 -1
- package/dist/platform-hap.js +0 -2861
- package/dist/platform-hap.js.map +0 -1
- package/dist/platform-matter.d.ts +0 -120
- package/dist/platform-matter.d.ts.map +0 -1
- package/dist/platform-matter.js +0 -966
- package/dist/platform-matter.js.map +0 -1
- package/dist/verifyconfig.test.d.ts +0 -2
- package/dist/verifyconfig.test.d.ts.map +0 -1
- package/dist/verifyconfig.test.js +0 -167
- package/dist/verifyconfig.test.js.map +0 -1
- package/src/custom.d.ts +0 -7
- package/src/devices-hap/airpurifier.ts +0 -563
- package/src/devices-hap/blindtilt.ts +0 -1049
- package/src/devices-hap/bot.ts +0 -900
- package/src/devices-hap/ceilinglight.ts +0 -742
- package/src/devices-hap/colorbulb.ts +0 -904
- package/src/devices-hap/contact.ts +0 -457
- package/src/devices-hap/curtain.ts +0 -944
- package/src/devices-hap/device.ts +0 -811
- package/src/devices-hap/fan.ts +0 -711
- package/src/devices-hap/hub.ts +0 -439
- package/src/devices-hap/humidifier.ts +0 -669
- package/src/devices-hap/iosensor.ts +0 -427
- package/src/devices-hap/lightstrip.ts +0 -836
- package/src/devices-hap/lock.ts +0 -620
- package/src/devices-hap/meter.ts +0 -426
- package/src/devices-hap/meterplus.ts +0 -430
- package/src/devices-hap/meterpro.ts +0 -522
- package/src/devices-hap/motion.ts +0 -390
- package/src/devices-hap/plug.ts +0 -423
- package/src/devices-hap/relayswitch.ts +0 -727
- package/src/devices-hap/robotvacuumcleaner.ts +0 -568
- package/src/devices-hap/waterdetector.ts +0 -400
- package/src/devices-matter/BaseMatterAccessory.ts +0 -131
- package/src/devices-matter/ColorLightAccessory.ts +0 -110
- package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -92
- package/src/devices-matter/ContactSensorAccessory.ts +0 -41
- package/src/devices-matter/DimmableLightAccessory.ts +0 -192
- package/src/devices-matter/DoorLockAccessory.ts +0 -60
- package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -123
- package/src/devices-matter/FanAccessory.ts +0 -95
- package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
- package/src/devices-matter/LeakSensorAccessory.ts +0 -40
- package/src/devices-matter/LightSensorAccessory.ts +0 -41
- package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
- package/src/devices-matter/OnOffLightAccessory.ts +0 -133
- package/src/devices-matter/OnOffOutletAccessory.ts +0 -46
- package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
- package/src/devices-matter/RoboticVacuumAccessory.ts +0 -407
- package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
- package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
- package/src/devices-matter/ThermostatAccessory.ts +0 -110
- package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
- package/src/devices-matter/WindowBlindAccessory.ts +0 -92
- package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
- package/src/devices-matter/custom/index.ts +0 -8
- package/src/devices-matter/index.ts +0 -29
- package/src/index.test.ts +0 -19
- package/src/irdevice/airconditioner.ts +0 -533
- package/src/irdevice/airpurifier.ts +0 -252
- package/src/irdevice/camera.ts +0 -129
- package/src/irdevice/fan.ts +0 -226
- package/src/irdevice/irdevice.ts +0 -344
- package/src/irdevice/light.ts +0 -246
- package/src/irdevice/other.ts +0 -790
- package/src/irdevice/tv.ts +0 -378
- package/src/irdevice/vacuumcleaner.ts +0 -126
- package/src/irdevice/waterheater.ts +0 -127
- package/src/platform-hap.ts +0 -2997
- package/src/platform-matter.ts +0 -1092
- package/src/verifyconfig.test.ts +0 -197
|
@@ -1,642 +0,0 @@
|
|
|
1
|
-
/*
|
|
2
|
-
* For Testing Locally:
|
|
3
|
-
* import { SwitchBotBLEModel, SwitchBotBLEModelName } from '/Users/Shared/GitHub/OpenWonderLabs/node-switchbot/dist/index.js';
|
|
4
|
-
*/
|
|
5
|
-
import { SwitchBotBLEModel, SwitchBotBLEModelName } from 'node-switchbot';
|
|
6
|
-
import { debounceTime, interval, skipWhile, Subject, take, tap } from 'rxjs';
|
|
7
|
-
import { formatDeviceIdAsMac } from '../utils.js';
|
|
8
|
-
import { deviceBase } from './device.js';
|
|
9
|
-
/**
|
|
10
|
-
* Platform Accessory
|
|
11
|
-
* An instance of this class is created for each accessory your platform registers
|
|
12
|
-
* Each accessory may expose multiple services of different service types.
|
|
13
|
-
*/
|
|
14
|
-
export class RelaySwitch extends deviceBase {
|
|
15
|
-
platform;
|
|
16
|
-
// Services
|
|
17
|
-
Switch;
|
|
18
|
-
GarageDoor;
|
|
19
|
-
Door;
|
|
20
|
-
Window;
|
|
21
|
-
WindowCovering;
|
|
22
|
-
LockMechanism;
|
|
23
|
-
Faucet;
|
|
24
|
-
Fan;
|
|
25
|
-
StatefulProgrammableSwitch;
|
|
26
|
-
Outlet;
|
|
27
|
-
On;
|
|
28
|
-
// OpenAPI
|
|
29
|
-
deviceStatus;
|
|
30
|
-
// Webhook
|
|
31
|
-
webhookContext;
|
|
32
|
-
// BLE
|
|
33
|
-
serviceData;
|
|
34
|
-
// Config
|
|
35
|
-
allowPush;
|
|
36
|
-
relaySwitchDeviceType;
|
|
37
|
-
// Updates
|
|
38
|
-
relaySwitchUpdateInProgress;
|
|
39
|
-
doRelaySwitchUpdate;
|
|
40
|
-
/**
|
|
41
|
-
* Constructs a new instance of the RelaySwitch device.
|
|
42
|
-
*
|
|
43
|
-
* @param {SwitchBotPlatform} platform - The platform instance.
|
|
44
|
-
* @param {PlatformAccessory} accessory - The platform accessory.
|
|
45
|
-
* @param {device & devicesConfig} device - The device configuration.
|
|
46
|
-
*
|
|
47
|
-
* Initializes the RelaySwitch device, sets up the battery service, maps the device type to the appropriate HomeKit service,
|
|
48
|
-
* removes unnecessary services, retrieves initial values, registers event handlers, and starts update intervals.
|
|
49
|
-
*
|
|
50
|
-
* @constructor
|
|
51
|
-
*/
|
|
52
|
-
constructor(platform, accessory, device) {
|
|
53
|
-
super(platform, accessory, device);
|
|
54
|
-
this.platform = platform;
|
|
55
|
-
this.getRelaySwitchConfigSettings(device);
|
|
56
|
-
// this is subject we use to track when we need to POST changes to the SwitchBot API
|
|
57
|
-
this.doRelaySwitchUpdate = new Subject();
|
|
58
|
-
this.relaySwitchUpdateInProgress = false;
|
|
59
|
-
/**
|
|
60
|
-
* A mapping of device types to their corresponding HomeKit categories, services, and characteristics.
|
|
61
|
-
*
|
|
62
|
-
* Each key represents a device type and maps to an object containing:
|
|
63
|
-
* - `category`: The HomeKit category for the device.
|
|
64
|
-
* - `service`: The HomeKit service associated with the device.
|
|
65
|
-
* - `characteristic`: The HomeKit characteristic for the device.
|
|
66
|
-
*/
|
|
67
|
-
const deviceTypeMap = {
|
|
68
|
-
switch: { category: 8 /* this.hap.Categories.SWITCH */, service: this.hap.Service.Switch, characteristic: this.hap.Characteristic.On },
|
|
69
|
-
garagedoor: { category: 4 /* this.hap.Categories.GARAGE_DOOR_OPENER */, service: this.hap.Service.GarageDoorOpener, characteristic: this.hap.Characteristic.TargetDoorState },
|
|
70
|
-
door: { category: 12 /* this.hap.Categories.DOOR */, service: this.hap.Service.Door, characteristic: this.hap.Characteristic.TargetPosition },
|
|
71
|
-
window: { category: 13 /* this.hap.Categories.WINDOW */, service: this.hap.Service.Window, characteristic: this.hap.Characteristic.TargetPosition },
|
|
72
|
-
windowcovering: { category: 14 /* this.hap.Categories.WINDOW_COVERING */, service: this.hap.Service.WindowCovering, characteristic: this.hap.Characteristic.TargetPosition },
|
|
73
|
-
lock: { category: 6 /* this.hap.Categories.DOOR_LOCK */, service: this.hap.Service.LockMechanism, characteristic: this.hap.Characteristic.LockTargetState },
|
|
74
|
-
faucet: { category: 29 /* this.hap.Categories.FAUCET */, service: this.hap.Service.Faucet, characteristic: this.hap.Characteristic.Active },
|
|
75
|
-
fan: { category: 3 /* this.hap.Categories.FAN */, service: this.hap.Service.Fanv2, characteristic: this.hap.Characteristic.Active },
|
|
76
|
-
stateful: { category: 15 /* this.hap.Categories.PROGRAMMABLE_SWITCH */, service: this.hap.Service.StatefulProgrammableSwitch, characteristic: this.hap.Characteristic.ProgrammableSwitchOutputState },
|
|
77
|
-
outlet: { category: 7 /* this.hap.Categories.OUTLET */, service: this.hap.Service.Outlet, characteristic: this.hap.Characteristic.On },
|
|
78
|
-
};
|
|
79
|
-
/**
|
|
80
|
-
* The type of the device, determined by mapping the relaySwitchDeviceType to a value in the deviceTypeMap.
|
|
81
|
-
*/
|
|
82
|
-
const deviceType = deviceTypeMap[this.relaySwitchDeviceType];
|
|
83
|
-
if (deviceType) {
|
|
84
|
-
// Set category
|
|
85
|
-
accessory.category = deviceType.category;
|
|
86
|
-
// Initialize Service
|
|
87
|
-
const contextKey = this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1);
|
|
88
|
-
accessory.context[contextKey] = accessory.context[contextKey] ?? {};
|
|
89
|
-
this[this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1)] = {
|
|
90
|
-
Name: accessory.displayName,
|
|
91
|
-
Service: accessory.getService(deviceType.service) ?? accessory.addService(deviceType.service),
|
|
92
|
-
};
|
|
93
|
-
accessory.context[contextKey] = this[this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1)];
|
|
94
|
-
this.debugLog(`Displaying as ${contextKey}`);
|
|
95
|
-
// Initialize Characteristics
|
|
96
|
-
this[this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1)].Service.setCharacteristic(this.hap.Characteristic.Name, this[this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1)].Name).getCharacteristic(deviceType.characteristic).onSet(this.OnSet.bind(this));
|
|
97
|
-
}
|
|
98
|
-
else {
|
|
99
|
-
this.errorLog('Device Type not set');
|
|
100
|
-
}
|
|
101
|
-
/**
|
|
102
|
-
* An array of service removal functions that should be executed based on the current state of the device.
|
|
103
|
-
* Each element in the array is a function that removes a specific service if the corresponding condition is met.
|
|
104
|
-
*/
|
|
105
|
-
const servicesToRemove = [
|
|
106
|
-
!this.StatefulProgrammableSwitch && this.removeStatefulProgrammableSwitchService,
|
|
107
|
-
!this.Outlet && this.removeOutletService,
|
|
108
|
-
!this.Window && this.removeWindowService,
|
|
109
|
-
!this.GarageDoor && this.removeGarageDoorService,
|
|
110
|
-
this.WindowCovering && this.removeWindowCoveringService,
|
|
111
|
-
!this.Switch && this.removeSwitchService,
|
|
112
|
-
!this.Faucet && this.removeFaucetService,
|
|
113
|
-
!this.Door && this.removeDoorService,
|
|
114
|
-
!this.LockMechanism && this.removeLockService,
|
|
115
|
-
!this.Fan && this.removeFanService,
|
|
116
|
-
].filter(Boolean);
|
|
117
|
-
for (const removeService of servicesToRemove) {
|
|
118
|
-
if (typeof removeService === 'function') {
|
|
119
|
-
removeService.call(this, accessory);
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
// Retrieve initial values and updateHomekit
|
|
123
|
-
try {
|
|
124
|
-
this.debugLog('Retrieve initial values and update Homekit');
|
|
125
|
-
this.refreshStatus();
|
|
126
|
-
}
|
|
127
|
-
catch (e) {
|
|
128
|
-
this.errorLog(`failed to retrieve initial values and update Homekit, Error: ${e.message ?? e}`);
|
|
129
|
-
}
|
|
130
|
-
// regisiter webhook event handler if enabled
|
|
131
|
-
try {
|
|
132
|
-
this.debugLog('Registering Webhook Event Handler');
|
|
133
|
-
this.registerWebhook();
|
|
134
|
-
}
|
|
135
|
-
catch (e) {
|
|
136
|
-
this.errorLog(`failed to registerWebhook, Error: ${e.message ?? e}`);
|
|
137
|
-
}
|
|
138
|
-
// regisiter platform BLE event handler if enabled
|
|
139
|
-
try {
|
|
140
|
-
this.debugLog('Registering Platform BLE Event Handler');
|
|
141
|
-
this.registerPlatformBLE();
|
|
142
|
-
}
|
|
143
|
-
catch (e) {
|
|
144
|
-
this.errorLog(`failed to registerPlatformBLE, Error: ${e.message ?? e}`);
|
|
145
|
-
}
|
|
146
|
-
// Start an update interval
|
|
147
|
-
interval(this.deviceRefreshRate * 1000)
|
|
148
|
-
.pipe(skipWhile(() => this.relaySwitchUpdateInProgress))
|
|
149
|
-
.subscribe(async () => {
|
|
150
|
-
await this.refreshStatus();
|
|
151
|
-
});
|
|
152
|
-
// Watch for RelaySwitch change events
|
|
153
|
-
// We put in a debounce of 1000ms so we don't make duplicate calls
|
|
154
|
-
this.doRelaySwitchUpdate
|
|
155
|
-
.pipe(tap(() => {
|
|
156
|
-
this.relaySwitchUpdateInProgress = true;
|
|
157
|
-
}), debounceTime(this.devicePushRate * 1000))
|
|
158
|
-
.subscribe(async () => {
|
|
159
|
-
try {
|
|
160
|
-
await this.pushChanges();
|
|
161
|
-
}
|
|
162
|
-
catch (e) {
|
|
163
|
-
await this.apiError(e);
|
|
164
|
-
this.errorLog(`failed pushChanges with ${device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`);
|
|
165
|
-
}
|
|
166
|
-
this.relaySwitchUpdateInProgress = false;
|
|
167
|
-
});
|
|
168
|
-
}
|
|
169
|
-
/**
|
|
170
|
-
* Parse the device status from the SwitchBotBLE API
|
|
171
|
-
*/
|
|
172
|
-
async BLEparseStatus() {
|
|
173
|
-
this.debugLog('BLEparseStatus');
|
|
174
|
-
this.debugLog(`(power, battery, deviceMode) = BLE:(${this.serviceData.state}), current:(${this.accessory.context.On})`);
|
|
175
|
-
// BLEmode (true if Switch Mode) | (false if Press Mode)
|
|
176
|
-
this.On = this.serviceData.mode ? this.serviceData.state : false;
|
|
177
|
-
const mode = this.serviceData.mode ? 'Switch' : 'Press';
|
|
178
|
-
this.debugLog(`${mode} Mode, On: ${this.On}`);
|
|
179
|
-
this.accessory.context.On = this.On;
|
|
180
|
-
}
|
|
181
|
-
/**
|
|
182
|
-
* Parse the device status from the SwitchBot OpenAPI
|
|
183
|
-
*/
|
|
184
|
-
async openAPIparseStatus() {
|
|
185
|
-
this.debugLog('openAPIparseStatus');
|
|
186
|
-
this.debugLog(`(power, battery, deviceMode) = API:(${this.deviceStatus.switchStatus}), current:(${this.accessory.context.On})`);
|
|
187
|
-
// On
|
|
188
|
-
this.On = this.deviceStatus.switchStatus !== 0;
|
|
189
|
-
this.debugLog(`On: ${this.On}`);
|
|
190
|
-
this.accessory.context.On = this.On;
|
|
191
|
-
// Firmware Version
|
|
192
|
-
if (this.deviceStatus.version) {
|
|
193
|
-
const version = this.deviceStatus.version;
|
|
194
|
-
this.debugLog(`Firmware Version: ${version.replace(/^V|-.*$/g, '')}`);
|
|
195
|
-
const deviceVersion = version.replace(/^V|-.*$/g, '') ?? '0.0.0';
|
|
196
|
-
this.accessory
|
|
197
|
-
.getService(this.hap.Service.AccessoryInformation)
|
|
198
|
-
.setCharacteristic(this.hap.Characteristic.HardwareRevision, deviceVersion)
|
|
199
|
-
.setCharacteristic(this.hap.Characteristic.FirmwareRevision, deviceVersion)
|
|
200
|
-
.getCharacteristic(this.hap.Characteristic.FirmwareRevision)
|
|
201
|
-
.updateValue(deviceVersion);
|
|
202
|
-
this.accessory.context.version = deviceVersion;
|
|
203
|
-
this.debugLog(`version: ${deviceVersion}`);
|
|
204
|
-
}
|
|
205
|
-
}
|
|
206
|
-
async parseStatusWebhook() {
|
|
207
|
-
this.debugLog('parseStatusWebhook');
|
|
208
|
-
this.debugLog(`(power, battery, deviceMode) = Webhook:(${this.webhookContext.switchStatus}), current:(${this.On})`);
|
|
209
|
-
// On
|
|
210
|
-
this.On = this.webhookContext.switchStatus !== 0;
|
|
211
|
-
this.debugLog(`On: ${this.On}`);
|
|
212
|
-
}
|
|
213
|
-
/**
|
|
214
|
-
* Asks the SwitchBot API for the latest device information
|
|
215
|
-
*/
|
|
216
|
-
async refreshStatus() {
|
|
217
|
-
if (this.BLE) {
|
|
218
|
-
await this.BLERefreshStatus();
|
|
219
|
-
}
|
|
220
|
-
else if (this.OpenAPI && this.platform.config.credentials?.token) {
|
|
221
|
-
await this.openAPIRefreshStatus();
|
|
222
|
-
}
|
|
223
|
-
else {
|
|
224
|
-
await this.offlineOff();
|
|
225
|
-
this.debugWarnLog(`Connection Type: ${this.device.connectionType}, refreshStatus will not happen.`);
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
async BLERefreshStatus() {
|
|
229
|
-
this.debugLog('BLERefreshStatus');
|
|
230
|
-
const switchBotBLE = await this.switchbotBLE();
|
|
231
|
-
if (switchBotBLE === undefined) {
|
|
232
|
-
await this.BLERefreshConnection(switchBotBLE);
|
|
233
|
-
}
|
|
234
|
-
else {
|
|
235
|
-
// Start to monitor advertisement packets
|
|
236
|
-
(async () => {
|
|
237
|
-
// Start to monitor advertisement packets
|
|
238
|
-
const serviceData = await this.monitorAdvertisementPackets(switchBotBLE);
|
|
239
|
-
// Update HomeKit
|
|
240
|
-
if ((serviceData.model === SwitchBotBLEModel.RelaySwitch1PM && serviceData.modelName === SwitchBotBLEModelName.RelaySwitch1PM) ?? (serviceData.model === SwitchBotBLEModel.RelaySwitch1 && serviceData.modelName === SwitchBotBLEModelName.RelaySwitch1)) {
|
|
241
|
-
this.serviceData = serviceData;
|
|
242
|
-
if (serviceData !== undefined || serviceData !== null) {
|
|
243
|
-
await this.BLEparseStatus();
|
|
244
|
-
await this.updateHomeKitCharacteristics();
|
|
245
|
-
}
|
|
246
|
-
else {
|
|
247
|
-
this.errorLog(`serviceData is either undefined or null, serviceData: ${JSON.stringify(serviceData)}`);
|
|
248
|
-
await this.BLERefreshConnection(switchBotBLE);
|
|
249
|
-
}
|
|
250
|
-
}
|
|
251
|
-
else {
|
|
252
|
-
this.errorLog(`failed to get serviceData, serviceData: ${JSON.stringify(serviceData)}`);
|
|
253
|
-
await this.BLERefreshConnection(switchBotBLE);
|
|
254
|
-
}
|
|
255
|
-
})();
|
|
256
|
-
}
|
|
257
|
-
}
|
|
258
|
-
async registerPlatformBLE() {
|
|
259
|
-
this.debugLog('registerPlatformBLE');
|
|
260
|
-
if (this.config.options?.BLE && !this.device.disablePlatformBLE) {
|
|
261
|
-
this.debugLog('is listening to Platform BLE.');
|
|
262
|
-
try {
|
|
263
|
-
const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId);
|
|
264
|
-
this.device.bleMac = formattedDeviceId;
|
|
265
|
-
this.debugLog(`bleMac: ${this.device.bleMac}`);
|
|
266
|
-
this.platform.bleEventHandler[this.device.bleMac] = async (context) => {
|
|
267
|
-
try {
|
|
268
|
-
this.serviceData = context;
|
|
269
|
-
if (context !== undefined || context !== null) {
|
|
270
|
-
this.debugLog(`received BLE: ${JSON.stringify(context)}`);
|
|
271
|
-
await this.BLEparseStatus();
|
|
272
|
-
await this.updateHomeKitCharacteristics();
|
|
273
|
-
}
|
|
274
|
-
else {
|
|
275
|
-
this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`);
|
|
276
|
-
await this.BLERefreshConnection(context);
|
|
277
|
-
}
|
|
278
|
-
}
|
|
279
|
-
catch (e) {
|
|
280
|
-
this.errorLog(`failed to handle BLE. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`);
|
|
281
|
-
}
|
|
282
|
-
};
|
|
283
|
-
}
|
|
284
|
-
catch (error) {
|
|
285
|
-
this.errorLog(`failed to format device ID as MAC, Error: ${error}`);
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
else {
|
|
289
|
-
this.debugLog('is not listening to Platform BLE');
|
|
290
|
-
}
|
|
291
|
-
}
|
|
292
|
-
async openAPIRefreshStatus() {
|
|
293
|
-
this.debugLog('openAPIRefreshStatus');
|
|
294
|
-
try {
|
|
295
|
-
const deviceStatus = await this.deviceRefreshStatus();
|
|
296
|
-
this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`);
|
|
297
|
-
if (await this.successfulStatusCodes(deviceStatus)) {
|
|
298
|
-
this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`);
|
|
299
|
-
this.deviceStatus = deviceStatus.body;
|
|
300
|
-
await this.openAPIparseStatus();
|
|
301
|
-
await this.updateHomeKitCharacteristics();
|
|
302
|
-
}
|
|
303
|
-
else {
|
|
304
|
-
this.debugWarnLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`);
|
|
305
|
-
}
|
|
306
|
-
}
|
|
307
|
-
catch (e) {
|
|
308
|
-
await this.apiError(e);
|
|
309
|
-
this.errorLog(`failed openAPIRefreshStatus with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`);
|
|
310
|
-
}
|
|
311
|
-
}
|
|
312
|
-
async registerWebhook() {
|
|
313
|
-
if (this.device.webhook) {
|
|
314
|
-
this.debugLog('is listening webhook.');
|
|
315
|
-
this.platform.webhookEventHandler[this.device.deviceId] = async (context) => {
|
|
316
|
-
try {
|
|
317
|
-
this.webhookContext = context;
|
|
318
|
-
if (context !== undefined || context !== null) {
|
|
319
|
-
this.debugLog(`received Webhook: ${JSON.stringify(context)}`);
|
|
320
|
-
await this.parseStatusWebhook();
|
|
321
|
-
await this.updateHomeKitCharacteristics();
|
|
322
|
-
}
|
|
323
|
-
else {
|
|
324
|
-
this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`);
|
|
325
|
-
}
|
|
326
|
-
}
|
|
327
|
-
catch (e) {
|
|
328
|
-
this.errorLog(`failed to handle webhook. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`);
|
|
329
|
-
}
|
|
330
|
-
};
|
|
331
|
-
}
|
|
332
|
-
else {
|
|
333
|
-
this.debugLog('is not listening webhook.');
|
|
334
|
-
}
|
|
335
|
-
}
|
|
336
|
-
/**
|
|
337
|
-
* Pushes the requested changes to the SwitchBot API
|
|
338
|
-
* deviceType commandType Command command parameter Description
|
|
339
|
-
* RelaySwitch "command" "turnOff" "default" = set to OFF state
|
|
340
|
-
* RelaySwitch "command" "turnOn" "default" = set to ON state
|
|
341
|
-
*/
|
|
342
|
-
async pushChanges() {
|
|
343
|
-
if (this.BLE) {
|
|
344
|
-
await this.BLEpushChanges();
|
|
345
|
-
}
|
|
346
|
-
else if (this.OpenAPI && this.platform.config.credentials?.token) {
|
|
347
|
-
await this.openAPIpushChanges();
|
|
348
|
-
}
|
|
349
|
-
else {
|
|
350
|
-
await this.offlineOff();
|
|
351
|
-
this.debugWarnLog(`Connection Type: ${this.device.connectionType}, pushChanges will not happen.`);
|
|
352
|
-
}
|
|
353
|
-
// Refresh the status from the API
|
|
354
|
-
interval(15000)
|
|
355
|
-
.pipe(skipWhile(() => this.relaySwitchUpdateInProgress))
|
|
356
|
-
.pipe(take(1))
|
|
357
|
-
.subscribe(async () => {
|
|
358
|
-
await this.refreshStatus();
|
|
359
|
-
});
|
|
360
|
-
}
|
|
361
|
-
async BLEpushChanges() {
|
|
362
|
-
this.debugLog('BLEpushChanges');
|
|
363
|
-
if ((this.On !== this.accessory.context.On) || this.allowPush) {
|
|
364
|
-
this.debugLog(`BLEpushChanges On: ${this.On} OnCached: ${this.accessory.context.On}`);
|
|
365
|
-
const switchBotBLE = this.platform.switchBotBLE;
|
|
366
|
-
try {
|
|
367
|
-
const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId);
|
|
368
|
-
this.device.bleMac = formattedDeviceId;
|
|
369
|
-
this.debugLog(`bleMac: ${this.device.bleMac}`);
|
|
370
|
-
switchBotBLE
|
|
371
|
-
.discover({ model: this.device.bleModel, quick: true, id: this.device.bleMac })
|
|
372
|
-
.then(async (device_list) => {
|
|
373
|
-
const deviceList = device_list;
|
|
374
|
-
this.infoLog(`On: ${this.On}`);
|
|
375
|
-
this.warnLog(`device_list: ${JSON.stringify(device_list)}`);
|
|
376
|
-
return await this.retryBLE({
|
|
377
|
-
max: this.maxRetryBLE(),
|
|
378
|
-
fn: async () => {
|
|
379
|
-
if (deviceList.length > 0) {
|
|
380
|
-
if (this.On) {
|
|
381
|
-
return await deviceList[0].turnOn();
|
|
382
|
-
}
|
|
383
|
-
else {
|
|
384
|
-
return await deviceList[0].turnOff();
|
|
385
|
-
}
|
|
386
|
-
}
|
|
387
|
-
else {
|
|
388
|
-
throw new Error('No device found');
|
|
389
|
-
}
|
|
390
|
-
},
|
|
391
|
-
});
|
|
392
|
-
})
|
|
393
|
-
.then(async () => {
|
|
394
|
-
this.successLog(`On: ${this.On} sent over SwitchBot BLE, sent successfully`);
|
|
395
|
-
await this.updateHomeKitCharacteristics();
|
|
396
|
-
})
|
|
397
|
-
.catch(async (e) => {
|
|
398
|
-
await this.apiError(e);
|
|
399
|
-
this.errorLog(`failed BLEpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`);
|
|
400
|
-
await this.BLEPushConnection();
|
|
401
|
-
});
|
|
402
|
-
}
|
|
403
|
-
catch (error) {
|
|
404
|
-
this.errorLog(`failed to format device ID as MAC, Error: ${error}`);
|
|
405
|
-
}
|
|
406
|
-
}
|
|
407
|
-
else {
|
|
408
|
-
this.debugLog(`No Changes (BLEpushChanges), On: ${this.On} OnCached: ${this.accessory.context.On}`);
|
|
409
|
-
}
|
|
410
|
-
}
|
|
411
|
-
async openAPIpushChanges() {
|
|
412
|
-
this.debugLog('openAPIpushChanges');
|
|
413
|
-
if ((this.On !== this.accessory.context.On) || this.allowPush) {
|
|
414
|
-
const bodyChange = {
|
|
415
|
-
command: this.On ? 'turnOn' : 'turnOff',
|
|
416
|
-
parameter: 'default',
|
|
417
|
-
commandType: 'command',
|
|
418
|
-
};
|
|
419
|
-
this.debugLog(`SwitchBot OpenAPI bodyChange: ${JSON.stringify(bodyChange)}`);
|
|
420
|
-
try {
|
|
421
|
-
const deviceStatus = await this.pushChangeRequest(bodyChange);
|
|
422
|
-
this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`);
|
|
423
|
-
if (await this.successfulStatusCodes(deviceStatus)) {
|
|
424
|
-
this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`);
|
|
425
|
-
await this.updateHomeKitCharacteristics();
|
|
426
|
-
}
|
|
427
|
-
else {
|
|
428
|
-
await this.statusCode(deviceStatus.statusCode);
|
|
429
|
-
}
|
|
430
|
-
}
|
|
431
|
-
catch (e) {
|
|
432
|
-
await this.apiError(e);
|
|
433
|
-
this.errorLog(`failed openAPIpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`);
|
|
434
|
-
}
|
|
435
|
-
}
|
|
436
|
-
else {
|
|
437
|
-
this.debugLog(`No Changes (openAPIpushChanges), On: ${this.On} OnCached: ${this.accessory.context.On}`);
|
|
438
|
-
}
|
|
439
|
-
}
|
|
440
|
-
/**
|
|
441
|
-
* Handle requests to set the "On" characteristic
|
|
442
|
-
*/
|
|
443
|
-
async OnSet(value) {
|
|
444
|
-
this.debugLog(`value: ${value}`);
|
|
445
|
-
const deviceTypeActions = {
|
|
446
|
-
switch: () => this.On = value !== false,
|
|
447
|
-
garagedoor: () => this.On = value !== this.hap.Characteristic.TargetDoorState.CLOSED,
|
|
448
|
-
door: () => this.On = value !== 0,
|
|
449
|
-
window: () => this.On = value !== 0,
|
|
450
|
-
windowcovering: () => this.On = value !== 0,
|
|
451
|
-
lock: () => this.On = value !== this.hap.Characteristic.LockTargetState.SECURED,
|
|
452
|
-
faucet: () => this.On = value !== this.hap.Characteristic.Active.INACTIVE,
|
|
453
|
-
fan: () => this.On = value !== 0,
|
|
454
|
-
stateful: () => this.On = value !== 0,
|
|
455
|
-
default: () => this.On = value !== false,
|
|
456
|
-
};
|
|
457
|
-
const action = deviceTypeActions[this.relaySwitchDeviceType] || deviceTypeActions.default;
|
|
458
|
-
action();
|
|
459
|
-
this.doRelaySwitchUpdate.next();
|
|
460
|
-
}
|
|
461
|
-
/**
|
|
462
|
-
* Updates the status for each of the HomeKit Characteristics
|
|
463
|
-
*/
|
|
464
|
-
async updateHomeKitCharacteristics() {
|
|
465
|
-
this.debugLog('updateHomeKitCharacteristics');
|
|
466
|
-
// State
|
|
467
|
-
const updateCharacteristic = async (service, characteristic, value, logMessage) => {
|
|
468
|
-
service.updateCharacteristic(characteristic, value);
|
|
469
|
-
this.debugLog(logMessage);
|
|
470
|
-
};
|
|
471
|
-
const stateActions = {
|
|
472
|
-
switch: async () => this.Switch && await updateCharacteristic(this.Switch.Service, this.hap.Characteristic.On, this.On, `updateCharacteristic On: ${this.On}`),
|
|
473
|
-
garagedoor: async () => {
|
|
474
|
-
if (this.GarageDoor) {
|
|
475
|
-
const targetState = this.On ? this.hap.Characteristic.TargetDoorState.OPEN : this.hap.Characteristic.TargetDoorState.CLOSED;
|
|
476
|
-
const currentState = this.On ? this.hap.Characteristic.CurrentDoorState.OPEN : this.hap.Characteristic.CurrentDoorState.CLOSED;
|
|
477
|
-
await updateCharacteristic(this.GarageDoor.Service, this.hap.Characteristic.TargetDoorState, targetState, `updateCharacteristic TargetDoorState: ${targetState}, CurrentDoorState: ${currentState} (${this.On})`);
|
|
478
|
-
await updateCharacteristic(this.GarageDoor.Service, this.hap.Characteristic.CurrentDoorState, currentState, '');
|
|
479
|
-
}
|
|
480
|
-
},
|
|
481
|
-
door: async () => {
|
|
482
|
-
if (this.Door) {
|
|
483
|
-
const position = this.On ? 100 : 0;
|
|
484
|
-
await updateCharacteristic(this.Door.Service, this.hap.Characteristic.TargetPosition, position, `updateCharacteristic TargetPosition: ${position}, CurrentPosition: ${position} (${this.On})`);
|
|
485
|
-
await updateCharacteristic(this.Door.Service, this.hap.Characteristic.CurrentPosition, position, '');
|
|
486
|
-
await updateCharacteristic(this.Door.Service, this.hap.Characteristic.PositionState, this.hap.Characteristic.PositionState.STOPPED, '');
|
|
487
|
-
}
|
|
488
|
-
},
|
|
489
|
-
window: async () => {
|
|
490
|
-
if (this.Window) {
|
|
491
|
-
const position = this.On ? 100 : 0;
|
|
492
|
-
await updateCharacteristic(this.Window.Service, this.hap.Characteristic.TargetPosition, position, `updateCharacteristic TargetPosition: ${position}, CurrentPosition: ${position} (${this.On})`);
|
|
493
|
-
await updateCharacteristic(this.Window.Service, this.hap.Characteristic.CurrentPosition, position, '');
|
|
494
|
-
await updateCharacteristic(this.Window.Service, this.hap.Characteristic.PositionState, this.hap.Characteristic.PositionState.STOPPED, '');
|
|
495
|
-
}
|
|
496
|
-
},
|
|
497
|
-
windowcovering: async () => {
|
|
498
|
-
if (this.WindowCovering) {
|
|
499
|
-
const position = this.On ? 100 : 0;
|
|
500
|
-
await updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.TargetPosition, position, `updateCharacteristic TargetPosition: ${position}, CurrentPosition: ${position} (${this.On})`);
|
|
501
|
-
await updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.CurrentPosition, position, '');
|
|
502
|
-
await updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.PositionState, this.hap.Characteristic.PositionState.STOPPED, '');
|
|
503
|
-
}
|
|
504
|
-
},
|
|
505
|
-
lock: async () => {
|
|
506
|
-
if (this.LockMechanism) {
|
|
507
|
-
const targetState = this.On ? this.hap.Characteristic.LockTargetState.UNSECURED : this.hap.Characteristic.LockTargetState.SECURED;
|
|
508
|
-
const currentState = this.On ? this.hap.Characteristic.LockCurrentState.UNSECURED : this.hap.Characteristic.LockCurrentState.SECURED;
|
|
509
|
-
await updateCharacteristic(this.LockMechanism.Service, this.hap.Characteristic.LockTargetState, targetState, `updateCharacteristic LockTargetState: ${targetState}, LockCurrentState: ${currentState} (${this.On})`);
|
|
510
|
-
await updateCharacteristic(this.LockMechanism.Service, this.hap.Characteristic.LockCurrentState, currentState, '');
|
|
511
|
-
}
|
|
512
|
-
},
|
|
513
|
-
faucet: async () => this.Faucet && await updateCharacteristic(this.Faucet.Service, this.hap.Characteristic.Active, this.On ? this.hap.Characteristic.Active.ACTIVE : this.hap.Characteristic.Active.INACTIVE, `updateCharacteristic Active: ${this.On}`),
|
|
514
|
-
fan: async () => this.Fan && await updateCharacteristic(this.Fan.Service, this.hap.Characteristic.Active, this.On ? this.hap.Characteristic.Active.ACTIVE : this.hap.Characteristic.Active.INACTIVE, `updateCharacteristic Active: ${this.On}`),
|
|
515
|
-
stateful: async () => {
|
|
516
|
-
if (this.StatefulProgrammableSwitch) {
|
|
517
|
-
await updateCharacteristic(this.StatefulProgrammableSwitch.Service, this.hap.Characteristic.ProgrammableSwitchEvent, this.hap.Characteristic.ProgrammableSwitchEvent.SINGLE_PRESS, `updateCharacteristic ProgrammableSwitchEvent: SINGLE_PRESS (${this.On})`);
|
|
518
|
-
await updateCharacteristic(this.StatefulProgrammableSwitch.Service, this.hap.Characteristic.ProgrammableSwitchOutputState, this.On ? 1 : 0, `updateCharacteristic ProgrammableSwitchOutputState: ${this.On ? 1 : 0}`);
|
|
519
|
-
}
|
|
520
|
-
},
|
|
521
|
-
outlet: async () => this.Outlet && await updateCharacteristic(this.Outlet.Service, this.hap.Characteristic.On, this.On, `updateCharacteristic On: ${this.On}`),
|
|
522
|
-
default: async () => this.errorLog(`relaySwitchDeviceType: ${this.relaySwitchDeviceType}, On: ${this.On}`),
|
|
523
|
-
};
|
|
524
|
-
const action = stateActions[this.relaySwitchDeviceType] || stateActions.default;
|
|
525
|
-
await action();
|
|
526
|
-
}
|
|
527
|
-
async removeService(accessory, serviceType, serviceName) {
|
|
528
|
-
const contextKey = serviceName.charAt(0).toUpperCase() + serviceName.slice(1);
|
|
529
|
-
accessory.context[contextKey] = accessory.context[contextKey] ?? {};
|
|
530
|
-
this[contextKey] = {
|
|
531
|
-
Name: accessory.context[contextKey].Name ?? accessory.displayName,
|
|
532
|
-
Service: accessory.getService(this.hap.Service[serviceType]),
|
|
533
|
-
};
|
|
534
|
-
accessory.context[contextKey] = this[contextKey];
|
|
535
|
-
this.debugWarnLog(`Removing any leftover ${contextKey} Service`);
|
|
536
|
-
accessory.removeService(this[contextKey].Service);
|
|
537
|
-
}
|
|
538
|
-
async removeOutletService(accessory) {
|
|
539
|
-
await this.removeService(accessory, 'Outlet', 'outlet');
|
|
540
|
-
}
|
|
541
|
-
async removeGarageDoorService(accessory) {
|
|
542
|
-
await this.removeService(accessory, 'GarageDoorOpener', 'garageDoor');
|
|
543
|
-
}
|
|
544
|
-
async removeDoorService(accessory) {
|
|
545
|
-
await this.removeService(accessory, 'Door', 'door');
|
|
546
|
-
}
|
|
547
|
-
async removeLockService(accessory) {
|
|
548
|
-
await this.removeService(accessory, 'LockMechanism', 'lockMechanism');
|
|
549
|
-
}
|
|
550
|
-
async removeFaucetService(accessory) {
|
|
551
|
-
await this.removeService(accessory, 'Valve', 'faucet');
|
|
552
|
-
}
|
|
553
|
-
async removeFanService(accessory) {
|
|
554
|
-
await this.removeService(accessory, 'Fan', 'fan');
|
|
555
|
-
}
|
|
556
|
-
async removeWindowService(accessory) {
|
|
557
|
-
await this.removeService(accessory, 'Window', 'window');
|
|
558
|
-
}
|
|
559
|
-
async removeWindowCoveringService(accessory) {
|
|
560
|
-
await this.removeService(accessory, 'WindowCovering', 'windowCovering');
|
|
561
|
-
}
|
|
562
|
-
async removeStatefulProgrammableSwitchService(accessory) {
|
|
563
|
-
await this.removeService(accessory, 'StatefulProgrammableSwitch', 'statefulProgrammableSwitch');
|
|
564
|
-
}
|
|
565
|
-
async removeSwitchService(accessory) {
|
|
566
|
-
await this.removeService(accessory, 'Switch', 'switch');
|
|
567
|
-
}
|
|
568
|
-
async getRelaySwitchConfigSettings(device) {
|
|
569
|
-
// RelaySwitch Device Type
|
|
570
|
-
this.relaySwitchDeviceType = device.type ?? 'Outlet';
|
|
571
|
-
const relaySwitchDeviceType = device.type ? 'Device Config' : 'Default';
|
|
572
|
-
this.debugWarnLog(`Use ${relaySwitchDeviceType} Device Type: ${this.relaySwitchDeviceType}`);
|
|
573
|
-
// RelaySwitch Allow Push
|
|
574
|
-
this.allowPush = device.allowPush ?? false;
|
|
575
|
-
const allowPush = device.allowPush
|
|
576
|
-
? `Using Allow Push: ${this.allowPush}`
|
|
577
|
-
: `No Allow Push Set, Using default Allow Push: ${this.allowPush}`;
|
|
578
|
-
this.debugWarnLog(allowPush);
|
|
579
|
-
}
|
|
580
|
-
async BLEPushConnection() {
|
|
581
|
-
if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
|
|
582
|
-
this.warnLog('Using OpenAPI Connection to Push Changes');
|
|
583
|
-
await this.openAPIpushChanges();
|
|
584
|
-
}
|
|
585
|
-
}
|
|
586
|
-
async BLERefreshConnection(switchbot) {
|
|
587
|
-
this.errorLog(`wasn't able to establish BLE Connection, node-switchbot: ${switchbot}`);
|
|
588
|
-
if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
|
|
589
|
-
this.warnLog('Using OpenAPI Connection to Refresh Status');
|
|
590
|
-
await this.openAPIRefreshStatus();
|
|
591
|
-
}
|
|
592
|
-
}
|
|
593
|
-
async offlineOff() {
|
|
594
|
-
if (this.device.offline) {
|
|
595
|
-
const updateCharacteristics = (service, characteristics) => {
|
|
596
|
-
for (const [characteristic, value] of Object.entries(characteristics)) {
|
|
597
|
-
service.updateCharacteristic(this.hap.Characteristic[characteristic], value);
|
|
598
|
-
}
|
|
599
|
-
};
|
|
600
|
-
const characteristicsMap = {
|
|
601
|
-
garagedoor: { TargetDoorState: 'CLOSED', CurrentDoorState: 'CLOSED', ObstructionDetected: false },
|
|
602
|
-
door: { TargetPosition: 0, CurrentPosition: 0, PositionState: 'STOPPED' },
|
|
603
|
-
window: { TargetPosition: 0, CurrentPosition: 0, PositionState: 'STOPPED' },
|
|
604
|
-
windowcovering: { TargetPosition: 0, CurrentPosition: 0, PositionState: 'STOPPED' },
|
|
605
|
-
lock: { LockTargetState: 'SECURED', LockCurrentState: 'SECURED' },
|
|
606
|
-
faucet: { Active: 'INACTIVE' },
|
|
607
|
-
fan: { On: false },
|
|
608
|
-
stateful: { ProgrammableSwitchEvent: 'SINGLE_PRESS', ProgrammableSwitchOutputState: 0 },
|
|
609
|
-
switch: { On: false },
|
|
610
|
-
outlet: { On: false },
|
|
611
|
-
};
|
|
612
|
-
const service = this[this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1)]?.Service;
|
|
613
|
-
if (service) {
|
|
614
|
-
updateCharacteristics(service, characteristicsMap[this.relaySwitchDeviceType]);
|
|
615
|
-
}
|
|
616
|
-
}
|
|
617
|
-
}
|
|
618
|
-
async apiError(e) {
|
|
619
|
-
const updateCharacteristics = (service, characteristics) => {
|
|
620
|
-
for (const [characteristic] of Object.entries(characteristics)) {
|
|
621
|
-
service.updateCharacteristic(this.hap.Characteristic[characteristic], e);
|
|
622
|
-
}
|
|
623
|
-
};
|
|
624
|
-
const characteristicsMap = {
|
|
625
|
-
garagedoor: { TargetDoorState: e, CurrentDoorState: e, ObstructionDetected: e },
|
|
626
|
-
door: { TargetPosition: e, CurrentPosition: e, PositionState: e },
|
|
627
|
-
window: { TargetPosition: e, CurrentPosition: e, PositionState: e },
|
|
628
|
-
windowcovering: { TargetPosition: e, CurrentPosition: e, PositionState: e },
|
|
629
|
-
lock: { LockTargetState: e, LockCurrentState: e },
|
|
630
|
-
faucet: { Active: e },
|
|
631
|
-
fan: { On: e },
|
|
632
|
-
stateful: { ProgrammableSwitchEvent: e, ProgrammableSwitchOutputState: e },
|
|
633
|
-
switch: { On: e },
|
|
634
|
-
outlet: { On: e },
|
|
635
|
-
};
|
|
636
|
-
const service = this[this.relaySwitchDeviceType.charAt(0).toUpperCase() + this.relaySwitchDeviceType.slice(1)]?.Service;
|
|
637
|
-
if (service) {
|
|
638
|
-
updateCharacteristics(service, characteristicsMap[this.relaySwitchDeviceType]);
|
|
639
|
-
}
|
|
640
|
-
}
|
|
641
|
-
}
|
|
642
|
-
//# sourceMappingURL=relayswitch.js.map
|