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

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 (271) hide show
  1. package/CHANGELOG.md +15 -0
  2. package/README.md +45 -3
  3. package/config.schema.json +871 -13754
  4. package/dist/devices-hap/airpurifier.d.ts.map +1 -1
  5. package/dist/devices-hap/airpurifier.js +12 -6
  6. package/dist/devices-hap/airpurifier.js.map +1 -1
  7. package/dist/devices-hap/blindtilt.js +3 -3
  8. package/dist/devices-hap/bot.d.ts.map +1 -1
  9. package/dist/devices-hap/bot.js +16 -5
  10. package/dist/devices-hap/bot.js.map +1 -1
  11. package/dist/devices-hap/ceilinglight.d.ts.map +1 -1
  12. package/dist/devices-hap/ceilinglight.js +13 -7
  13. package/dist/devices-hap/ceilinglight.js.map +1 -1
  14. package/dist/devices-hap/colorbulb.d.ts.map +1 -1
  15. package/dist/devices-hap/colorbulb.js +49 -9
  16. package/dist/devices-hap/colorbulb.js.map +1 -1
  17. package/dist/devices-hap/contact.js +3 -3
  18. package/dist/devices-hap/curtain.js +2 -2
  19. package/dist/devices-hap/curtain.js.map +1 -1
  20. package/dist/devices-hap/device.d.ts +18 -8
  21. package/dist/devices-hap/device.d.ts.map +1 -1
  22. package/dist/devices-hap/device.js +141 -69
  23. package/dist/devices-hap/device.js.map +1 -1
  24. package/dist/devices-hap/fan.d.ts.map +1 -1
  25. package/dist/devices-hap/fan.js +12 -6
  26. package/dist/devices-hap/fan.js.map +1 -1
  27. package/dist/devices-hap/hub.d.ts.map +1 -1
  28. package/dist/devices-hap/hub.js +6 -5
  29. package/dist/devices-hap/hub.js.map +1 -1
  30. package/dist/devices-hap/humidifier.d.ts +5 -0
  31. package/dist/devices-hap/humidifier.d.ts.map +1 -1
  32. package/dist/devices-hap/humidifier.js +92 -4
  33. package/dist/devices-hap/humidifier.js.map +1 -1
  34. package/dist/devices-hap/iosensor.d.ts.map +1 -1
  35. package/dist/devices-hap/iosensor.js +36 -21
  36. package/dist/devices-hap/iosensor.js.map +1 -1
  37. package/dist/devices-hap/lightstrip.d.ts.map +1 -1
  38. package/dist/devices-hap/lightstrip.js +38 -8
  39. package/dist/devices-hap/lightstrip.js.map +1 -1
  40. package/dist/devices-hap/lock.d.ts.map +1 -1
  41. package/dist/devices-hap/lock.js +14 -6
  42. package/dist/devices-hap/lock.js.map +1 -1
  43. package/dist/devices-hap/meter.d.ts.map +1 -1
  44. package/dist/devices-hap/meter.js +6 -5
  45. package/dist/devices-hap/meter.js.map +1 -1
  46. package/dist/devices-hap/meterplus.d.ts.map +1 -1
  47. package/dist/devices-hap/meterplus.js +6 -5
  48. package/dist/devices-hap/meterplus.js.map +1 -1
  49. package/dist/devices-hap/meterpro.d.ts.map +1 -1
  50. package/dist/devices-hap/meterpro.js +7 -6
  51. package/dist/devices-hap/meterpro.js.map +1 -1
  52. package/dist/devices-hap/motion.js +3 -3
  53. package/dist/devices-hap/plug.d.ts.map +1 -1
  54. package/dist/devices-hap/plug.js +11 -6
  55. package/dist/devices-hap/plug.js.map +1 -1
  56. package/dist/devices-hap/relayswitch.js +3 -3
  57. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +1 -1
  58. package/dist/devices-hap/robotvacuumcleaner.js +13 -6
  59. package/dist/devices-hap/robotvacuumcleaner.js.map +1 -1
  60. package/dist/devices-hap/waterdetector.js +3 -3
  61. package/dist/devices-matter/BaseMatterAccessory.d.ts +27 -0
  62. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +1 -1
  63. package/dist/devices-matter/BaseMatterAccessory.js +169 -5
  64. package/dist/devices-matter/BaseMatterAccessory.js.map +1 -1
  65. package/dist/devices-matter/ColorLightAccessory.d.ts.map +1 -1
  66. package/dist/devices-matter/ColorLightAccessory.js +12 -12
  67. package/dist/devices-matter/ColorLightAccessory.js.map +1 -1
  68. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +1 -1
  69. package/dist/devices-matter/ColorTemperatureLightAccessory.js +5 -7
  70. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +1 -1
  71. package/dist/devices-matter/DimmableLightAccessory.js +9 -9
  72. package/dist/devices-matter/DimmableLightAccessory.js.map +1 -1
  73. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +1 -1
  74. package/dist/devices-matter/ExtendedColorLightAccessory.js +14 -15
  75. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +1 -1
  76. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +1 -1
  77. package/dist/devices-matter/OnOffLightAccessory.js +8 -16
  78. package/dist/devices-matter/OnOffLightAccessory.js.map +1 -1
  79. package/dist/devices-matter/OnOffOutletAccessory.d.ts +2 -0
  80. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +1 -1
  81. package/dist/devices-matter/OnOffOutletAccessory.js +10 -7
  82. package/dist/devices-matter/OnOffOutletAccessory.js.map +1 -1
  83. package/dist/devices-matter/OnOffSwitchAccessory.js +2 -2
  84. package/dist/devices-matter/OnOffSwitchAccessory.js.map +1 -1
  85. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +36 -43
  86. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +1 -1
  87. package/dist/devices-matter/RoboticVacuumAccessory.js +478 -268
  88. package/dist/devices-matter/RoboticVacuumAccessory.js.map +1 -1
  89. package/dist/devices-matter/VenetianBlindAccessory.d.ts +6 -6
  90. package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +1 -1
  91. package/dist/devices-matter/VenetianBlindAccessory.js.map +1 -1
  92. package/dist/devices-matter/WindowBlindAccessory.d.ts +5 -5
  93. package/dist/devices-matter/WindowBlindAccessory.d.ts.map +1 -1
  94. package/dist/devices-matter/WindowBlindAccessory.js +57 -6
  95. package/dist/devices-matter/WindowBlindAccessory.js.map +1 -1
  96. package/dist/homebridge-ui/public/index.html +219 -19
  97. package/dist/homebridge-ui/server.js +0 -31
  98. package/dist/homebridge-ui/server.js.map +1 -1
  99. package/dist/index.d.ts.map +1 -1
  100. package/dist/index.js +4 -9
  101. package/dist/index.js.map +1 -1
  102. package/dist/irdevice/irdevice.d.ts +11 -10
  103. package/dist/irdevice/irdevice.d.ts.map +1 -1
  104. package/dist/irdevice/irdevice.js +76 -35
  105. package/dist/irdevice/irdevice.js.map +1 -1
  106. package/dist/platform-hap.d.ts +26 -15
  107. package/dist/platform-hap.d.ts.map +1 -1
  108. package/dist/platform-hap.js +333 -153
  109. package/dist/platform-hap.js.map +1 -1
  110. package/dist/platform-matter.d.ts +93 -6
  111. package/dist/platform-matter.d.ts.map +1 -1
  112. package/dist/platform-matter.js +1878 -229
  113. package/dist/platform-matter.js.map +1 -1
  114. package/dist/settings.d.ts +75 -7
  115. package/dist/settings.d.ts.map +1 -1
  116. package/dist/settings.js.map +1 -1
  117. package/dist/test/apiRequestTracker.test.d.ts +2 -0
  118. package/dist/test/apiRequestTracker.test.d.ts.map +1 -0
  119. package/dist/test/apiRequestTracker.test.js +392 -0
  120. package/dist/test/apiRequestTracker.test.js.map +1 -0
  121. package/dist/test/hap/device-webhook-context.test.d.ts +2 -0
  122. package/dist/test/hap/device-webhook-context.test.d.ts.map +1 -0
  123. package/dist/test/hap/device-webhook-context.test.js +128 -0
  124. package/dist/test/hap/device-webhook-context.test.js.map +1 -0
  125. package/dist/test/hap/platform-hap.logging.test.d.ts +2 -0
  126. package/dist/test/hap/platform-hap.logging.test.d.ts.map +1 -0
  127. package/dist/test/hap/platform-hap.logging.test.js +33 -0
  128. package/dist/test/hap/platform-hap.logging.test.js.map +1 -0
  129. package/dist/test/hap/platform-hap.test.d.ts +2 -0
  130. package/dist/test/hap/platform-hap.test.d.ts.map +1 -0
  131. package/dist/test/hap/platform-hap.test.js +62 -0
  132. package/dist/test/hap/platform-hap.test.js.map +1 -0
  133. package/dist/test/helpers/platform-fixtures.d.ts +9 -0
  134. package/dist/test/helpers/platform-fixtures.d.ts.map +1 -0
  135. package/dist/test/helpers/platform-fixtures.js +30 -0
  136. package/dist/test/helpers/platform-fixtures.js.map +1 -0
  137. package/dist/test/homebridge-ui/server.test.d.ts +2 -0
  138. package/dist/test/homebridge-ui/server.test.d.ts.map +1 -0
  139. package/dist/test/homebridge-ui/server.test.js +445 -0
  140. package/dist/test/homebridge-ui/server.test.js.map +1 -0
  141. package/dist/{index.test.d.ts.map → test/index.test.d.ts.map} +1 -1
  142. package/dist/test/index.test.js +19 -0
  143. package/dist/test/index.test.js.map +1 -0
  144. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts +2 -0
  145. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts.map +1 -0
  146. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js +71 -0
  147. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js.map +1 -0
  148. package/dist/test/matter/platform-matter.additional.test.d.ts +2 -0
  149. package/dist/test/matter/platform-matter.additional.test.d.ts.map +1 -0
  150. package/dist/test/matter/platform-matter.additional.test.js +35 -0
  151. package/dist/test/matter/platform-matter.additional.test.js.map +1 -0
  152. package/dist/test/matter/platform-matter.bleparse.test.d.ts +2 -0
  153. package/dist/test/matter/platform-matter.bleparse.test.d.ts.map +1 -0
  154. package/dist/test/matter/platform-matter.bleparse.test.js +43 -0
  155. package/dist/test/matter/platform-matter.bleparse.test.js.map +1 -0
  156. package/dist/test/matter/platform-matter.cleanup.test.d.ts +2 -0
  157. package/dist/test/matter/platform-matter.cleanup.test.d.ts.map +1 -0
  158. package/dist/test/matter/platform-matter.cleanup.test.js +70 -0
  159. package/dist/test/matter/platform-matter.cleanup.test.js.map +1 -0
  160. package/dist/test/matter/platform-matter.keepstale.test.d.ts +2 -0
  161. package/dist/test/matter/platform-matter.keepstale.test.d.ts.map +1 -0
  162. package/dist/test/matter/platform-matter.keepstale.test.js +27 -0
  163. package/dist/test/matter/platform-matter.keepstale.test.js.map +1 -0
  164. package/dist/test/matter/platform-matter.logging.test.d.ts +2 -0
  165. package/dist/test/matter/platform-matter.logging.test.d.ts.map +1 -0
  166. package/dist/test/matter/platform-matter.logging.test.js +29 -0
  167. package/dist/test/matter/platform-matter.logging.test.js.map +1 -0
  168. package/dist/test/matter/platform-matter.mapping.test.d.ts +2 -0
  169. package/dist/test/matter/platform-matter.mapping.test.d.ts.map +1 -0
  170. package/dist/test/matter/platform-matter.mapping.test.js +43 -0
  171. package/dist/test/matter/platform-matter.mapping.test.js.map +1 -0
  172. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts +2 -0
  173. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts.map +1 -0
  174. package/dist/test/matter/platform-matter.openapi-mapping.test.js +84 -0
  175. package/dist/test/matter/platform-matter.openapi-mapping.test.js.map +1 -0
  176. package/dist/test/matter/platform-matter.test.d.ts +2 -0
  177. package/dist/test/matter/platform-matter.test.d.ts.map +1 -0
  178. package/dist/test/matter/platform-matter.test.js +117 -0
  179. package/dist/test/matter/platform-matter.test.js.map +1 -0
  180. package/dist/test/matter/platform-matter.unregister.test.d.ts +2 -0
  181. package/dist/test/matter/platform-matter.unregister.test.d.ts.map +1 -0
  182. package/dist/test/matter/platform-matter.unregister.test.js +30 -0
  183. package/dist/test/matter/platform-matter.unregister.test.js.map +1 -0
  184. package/dist/test/matter/platform-matter.webhook.test.d.ts +2 -0
  185. package/dist/test/matter/platform-matter.webhook.test.d.ts.map +1 -0
  186. package/dist/test/matter/platform-matter.webhook.test.js +46 -0
  187. package/dist/test/matter/platform-matter.webhook.test.js.map +1 -0
  188. package/dist/test/utils.test.d.ts +2 -0
  189. package/dist/test/utils.test.d.ts.map +1 -0
  190. package/dist/test/utils.test.js +95 -0
  191. package/dist/test/utils.test.js.map +1 -0
  192. package/dist/test/verifyconfig.test.d.ts.map +1 -0
  193. package/dist/{verifyconfig.test.js → test/verifyconfig.test.js} +2 -2
  194. package/dist/test/verifyconfig.test.js.map +1 -0
  195. package/dist/utils.d.ts +204 -3
  196. package/dist/utils.d.ts.map +1 -1
  197. package/dist/utils.js +713 -33
  198. package/dist/utils.js.map +1 -1
  199. package/docs/assets/highlight.css +14 -0
  200. package/docs/assets/main.js +2 -2
  201. package/docs/index.html +31 -2
  202. package/docs/variables/default.html +1 -1
  203. package/package.json +15 -15
  204. package/src/devices-hap/airpurifier.ts +11 -6
  205. package/src/devices-hap/blindtilt.ts +3 -3
  206. package/src/devices-hap/bot.ts +15 -5
  207. package/src/devices-hap/ceilinglight.ts +12 -7
  208. package/src/devices-hap/colorbulb.ts +46 -10
  209. package/src/devices-hap/contact.ts +3 -3
  210. package/src/devices-hap/curtain.ts +2 -2
  211. package/src/devices-hap/device.ts +149 -70
  212. package/src/devices-hap/fan.ts +11 -6
  213. package/src/devices-hap/hub.ts +6 -5
  214. package/src/devices-hap/humidifier.ts +97 -4
  215. package/src/devices-hap/iosensor.ts +36 -21
  216. package/src/devices-hap/lightstrip.ts +35 -8
  217. package/src/devices-hap/lock.ts +13 -6
  218. package/src/devices-hap/meter.ts +6 -5
  219. package/src/devices-hap/meterplus.ts +6 -5
  220. package/src/devices-hap/meterpro.ts +7 -6
  221. package/src/devices-hap/motion.ts +3 -3
  222. package/src/devices-hap/plug.ts +10 -6
  223. package/src/devices-hap/relayswitch.ts +3 -3
  224. package/src/devices-hap/robotvacuumcleaner.ts +12 -6
  225. package/src/devices-hap/waterdetector.ts +3 -3
  226. package/src/devices-matter/BaseMatterAccessory.ts +176 -5
  227. package/src/devices-matter/ColorLightAccessory.ts +12 -12
  228. package/src/devices-matter/ColorTemperatureLightAccessory.ts +5 -7
  229. package/src/devices-matter/DimmableLightAccessory.ts +9 -9
  230. package/src/devices-matter/ExtendedColorLightAccessory.ts +14 -15
  231. package/src/devices-matter/OnOffLightAccessory.ts +8 -16
  232. package/src/devices-matter/OnOffOutletAccessory.ts +12 -7
  233. package/src/devices-matter/OnOffSwitchAccessory.ts +2 -2
  234. package/src/devices-matter/RoboticVacuumAccessory.ts +529 -315
  235. package/src/devices-matter/VenetianBlindAccessory.ts +5 -5
  236. package/src/devices-matter/WindowBlindAccessory.ts +53 -10
  237. package/src/homebridge-ui/public/index.html +219 -19
  238. package/src/homebridge-ui/server.ts +0 -34
  239. package/src/index.ts +4 -10
  240. package/src/irdevice/irdevice.ts +74 -35
  241. package/src/platform-hap.ts +365 -169
  242. package/src/platform-matter.ts +1923 -230
  243. package/src/settings.ts +78 -3
  244. package/src/test/apiRequestTracker.test.ts +417 -0
  245. package/src/test/hap/device-webhook-context.test.ts +136 -0
  246. package/src/test/hap/platform-hap.logging.test.ts +36 -0
  247. package/src/test/hap/platform-hap.test.ts +70 -0
  248. package/src/test/helpers/platform-fixtures.ts +33 -0
  249. package/src/test/homebridge-ui/server.test.ts +486 -0
  250. package/src/test/index.test.ts +24 -0
  251. package/src/test/matter/devices-matter/baseMatterAccessory.test.ts +88 -0
  252. package/src/test/matter/platform-matter.additional.test.ts +44 -0
  253. package/src/test/matter/platform-matter.bleparse.test.ts +47 -0
  254. package/src/test/matter/platform-matter.cleanup.test.ts +86 -0
  255. package/src/test/matter/platform-matter.keepstale.test.ts +37 -0
  256. package/src/test/matter/platform-matter.logging.test.ts +33 -0
  257. package/src/test/matter/platform-matter.mapping.test.ts +57 -0
  258. package/src/test/matter/platform-matter.openapi-mapping.test.ts +109 -0
  259. package/src/test/matter/platform-matter.test.ts +144 -0
  260. package/src/test/matter/platform-matter.unregister.test.ts +39 -0
  261. package/src/test/matter/platform-matter.webhook.test.ts +54 -0
  262. package/src/test/utils.test.ts +96 -0
  263. package/src/{verifyconfig.test.ts → test/verifyconfig.test.ts} +12 -11
  264. package/src/utils.ts +777 -36
  265. package/dist/index.test.js +0 -14
  266. package/dist/index.test.js.map +0 -1
  267. package/dist/verifyconfig.test.d.ts.map +0 -1
  268. package/dist/verifyconfig.test.js.map +0 -1
  269. package/src/index.test.ts +0 -19
  270. /package/dist/{index.test.d.ts → test/index.test.d.ts} +0 -0
  271. /package/dist/{verifyconfig.test.d.ts → test/verifyconfig.test.d.ts} +0 -0
package/src/settings.ts CHANGED
@@ -32,10 +32,18 @@ interface credentials {
32
32
 
33
33
  export interface options {
34
34
  devices?: devicesConfig[]
35
- deviceConfig?: { [deviceType: string]: devicesConfig }
36
35
  irdevices?: irDevicesConfig[]
37
- irdeviceConfig?: { [remoteType: string]: irDevicesConfig }
38
36
  allowInvalidCharacters?: boolean
37
+ // When true, devices declared in config.options.devices that are not
38
+ // discovered via the SwitchBot OpenAPI will still be included (config-only
39
+ // devices). Default: false.
40
+ allowConfigOnlyDevices?: boolean
41
+ /**
42
+ * When true, previously-registered accessories for devices that are no
43
+ * longer discovered or configured will be kept on the bridge. Default: false.
44
+ * When false (default), stale accessories are removed automatically.
45
+ */
46
+ keepStaleAccessories?: boolean
39
47
  mqttURL?: string
40
48
  mqttOptions?: IClientOptions
41
49
  mqttPubOptions?: IClientOptions
@@ -45,12 +53,62 @@ export interface options {
45
53
  disableLogsforOpenAPI?: boolean
46
54
  hostname?: string
47
55
  webhookURL?: string
56
+ /**
57
+ * When true, enables webhook support for all devices by default.
58
+ * Individual devices can override this with their own webhook setting.
59
+ * Requires webhookURL to be configured.
60
+ */
61
+ webhook?: boolean
48
62
  maxRetries?: number
49
63
  delayBetweenRetries?: number
50
64
  refreshRate?: number
51
65
  updateRate?: number
52
66
  pushRate?: number
53
67
  logging?: string
68
+ /**
69
+ * Maximum number of SwitchBot OpenAPI requests allowed per day.
70
+ * Defaults to 10,000 if not specified.
71
+ */
72
+ dailyApiLimit?: number
73
+ /**
74
+ * Number of daily API requests reserved for user-initiated commands.
75
+ * When remaining budget falls below this reserve, background polling and discovery
76
+ * are paused until the daily counter resets. Defaults to 1,000.
77
+ */
78
+ dailyApiReserveForCommands?: number
79
+ /**
80
+ * When true, the plugin will completely stop background polling/discovery
81
+ * once the remaining daily budget reaches the reserve (webhook-only mode).
82
+ * When false, polling continues until the hard daily limit is reached.
83
+ * Default: false.
84
+ */
85
+ webhookOnlyOnReserve?: boolean
86
+ /**
87
+ * When true, reset the daily API request counter at LOCAL midnight (system timezone).
88
+ * When false (default), reset at UTC midnight. Default: false.
89
+ */
90
+ dailyApiResetAtLocalMidnight?: boolean
91
+ /**
92
+ * When true, resets the daily API request counter to zero. This is useful for testing purposes.
93
+ */
94
+ resetDailyApiCounter?: boolean
95
+ /**
96
+ * When set, configures the batch refresh rate (in milliseconds) for Matter devices. Default: 5000 ms.
97
+ */
98
+ matterBatchRefreshRate?: number
99
+ /**
100
+ * When true, enables batch processing for Matter devices. Default: false.
101
+ */
102
+ matterBatchConcurrency?: number
103
+ /**
104
+ * When true, enables batch processing for Matter devices. Default: false.
105
+ */
106
+ matterBatchEnabled?: boolean
107
+ /**
108
+ * When set, adds a random delay (jitter) to batch requests to avoid thundering herd problems.
109
+ */
110
+ matterBatchJitter?: number
111
+ newFeatureEnabled?: boolean
54
112
  };
55
113
 
56
114
  export type devicesConfig = botConfig | relaySwitch1Config | relaySwitch1PMConfig | meterConfig | meterProConfig | indoorOutdoorSensorConfig | humidifierConfig | curtainConfig | blindTiltConfig | contactConfig | motionConfig | waterDetectorConfig | plugConfig | colorBulbConfig | stripLightConfig | ceilingLightConfig | lockConfig | hubConfig
@@ -85,6 +143,12 @@ export interface BaseDeviceConfig extends device {
85
143
  mqttPubOptions?: IClientOptions
86
144
  history?: boolean
87
145
  webhook?: boolean
146
+ /**
147
+ * When true, applies this device's configuration to all other devices
148
+ * of the same deviceType/configDeviceType (e.g., all Humidifiers).
149
+ * Specific per-device settings will override these template settings.
150
+ */
151
+ applyToAllDevicesOfType?: boolean
88
152
  }
89
153
 
90
154
  export interface botConfig extends BaseDeviceConfig {
@@ -136,6 +200,11 @@ export interface humidifierConfig extends BaseDeviceConfig {
136
200
  hide_temperature?: boolean
137
201
  convertUnitTo?: string
138
202
  set_minStep?: number
203
+ /**
204
+ * When true (Humidifier2 only), exposes a Switch service in HomeKit
205
+ * to trigger the built-in Drying Filter mode via OpenAPI (setMode 8).
206
+ */
207
+ activate_dryingfilter?: boolean
139
208
  };
140
209
 
141
210
  export interface curtainConfig extends BaseDeviceConfig {
@@ -211,7 +280,7 @@ export interface ceilingLightConfig extends BaseDeviceConfig {
211
280
  };
212
281
 
213
282
  export interface lockConfig extends BaseDeviceConfig {
214
- configDeviceType: 'Smart Lock' | 'Smart Lock Pro'
283
+ configDeviceType: 'Smart Lock' | 'Smart Lock Pro' | 'Smart Lock Ultra'
215
284
  hide_contactsensor?: boolean
216
285
  activate_latchbutton?: boolean
217
286
  };
@@ -244,6 +313,12 @@ export interface irBaseDeviceConfig extends irdevice {
244
313
  logging?: string
245
314
  customOn?: string
246
315
  customOff?: string
316
+ /**
317
+ * When true, applies this IR device's configuration to all other IR devices
318
+ * of the same remoteType/configRemoteType (e.g., all IR Fans).
319
+ * Specific per-device settings will override these template settings.
320
+ */
321
+ applyToAllDevicesOfType?: boolean
247
322
  customize?: boolean
248
323
  commandType?: string
249
324
  disablePushOn?: boolean
@@ -0,0 +1,417 @@
1
+ import type { API, Logging } from 'homebridge'
2
+
3
+ import { existsSync, mkdirSync, readdirSync, readFileSync, rmdirSync, unlinkSync } from 'node:fs'
4
+ import { tmpdir } from 'node:os'
5
+ import { join } from 'node:path'
6
+
7
+ import { describe, expect, it, vi } from 'vitest'
8
+
9
+ import { ApiRequestTracker } from '../utils.js'
10
+
11
+ // Helper to create isolated test environment for each test
12
+ function createTestEnvironment(pluginName = 'SwitchBotTest') {
13
+ const testId = Math.random().toString(36).substring(7)
14
+ const testDir = join(tmpdir(), `switchbot-test-${testId}`)
15
+
16
+ // Create test directory
17
+ if (!existsSync(testDir)) {
18
+ mkdirSync(testDir, { recursive: true })
19
+ }
20
+
21
+ const testStatsFile = join(testDir, `${pluginName.toLowerCase()}-api-stats.json`)
22
+
23
+ // Mock API with a unique storage path per test
24
+ const mockApi = {
25
+ user: {
26
+ storagePath: () => testDir,
27
+ },
28
+ } as unknown as API
29
+
30
+ // Mock logger
31
+ const mockLog = {
32
+ info: vi.fn(),
33
+ warn: vi.fn(),
34
+ error: vi.fn(),
35
+ debug: vi.fn(),
36
+ } as unknown as Logging
37
+
38
+ return { mockApi, mockLog, testStatsFile, testDir }
39
+ }
40
+
41
+ // Cleanup helper
42
+ function cleanup(testDir: string) {
43
+ try {
44
+ if (existsSync(testDir)) {
45
+ const files = readdirSync(testDir)
46
+ for (const file of files) {
47
+ try {
48
+ unlinkSync(join(testDir, file))
49
+ } catch {
50
+ // ignore
51
+ }
52
+ }
53
+ rmdirSync(testDir)
54
+ }
55
+ } catch {
56
+ // ignore
57
+ }
58
+ }
59
+
60
+ describe('apiRequestTracker', () => {
61
+ describe('initialization', () => {
62
+ it('should create a new tracker with default limits', () => {
63
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
64
+ try {
65
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
66
+ expect(tracker).toBeDefined()
67
+ expect(tracker.getCount()).toBe(0)
68
+ expect(tracker.getDate()).toBe(new Date().toISOString().split('T')[0])
69
+ } finally {
70
+ cleanup(testDir)
71
+ }
72
+ })
73
+
74
+ it('should respect custom daily limit', () => {
75
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
76
+ try {
77
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
78
+ dailyLimit: 5000,
79
+ reserveForCommands: 500,
80
+ })
81
+ expect(tracker).toBeDefined()
82
+ } finally {
83
+ cleanup(testDir)
84
+ }
85
+ })
86
+
87
+ it('should load existing stats from file', () => {
88
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
89
+ try {
90
+ // Create a tracker, increment, and verify persistence
91
+ const tracker1 = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
92
+ tracker1.track()
93
+ tracker1.track()
94
+ expect(tracker1.getCount()).toBe(2)
95
+
96
+ // Create a new tracker instance and verify it loads the count
97
+ const tracker2 = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
98
+ expect(tracker2.getCount()).toBe(2)
99
+ } finally {
100
+ cleanup(testDir)
101
+ }
102
+ })
103
+ })
104
+
105
+ describe('track() - legacy method', () => {
106
+ it('should increment the counter', () => {
107
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
108
+ try {
109
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
110
+ expect(tracker.getCount()).toBe(0)
111
+ tracker.track()
112
+ expect(tracker.getCount()).toBe(1)
113
+ tracker.track()
114
+ expect(tracker.getCount()).toBe(2)
115
+ } finally {
116
+ cleanup(testDir)
117
+ }
118
+ })
119
+
120
+ it('should persist count to file', () => {
121
+ const { mockApi, mockLog, testStatsFile, testDir } = createTestEnvironment()
122
+ try {
123
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
124
+ tracker.track()
125
+ tracker.track()
126
+ tracker.track()
127
+
128
+ // Read the stats file directly
129
+ const statsContent = readFileSync(testStatsFile, 'utf8')
130
+ const stats = JSON.parse(statsContent)
131
+ expect(stats.count).toBe(3)
132
+ expect(stats.date).toBe(new Date().toISOString().split('T')[0])
133
+ } finally {
134
+ cleanup(testDir)
135
+ }
136
+ })
137
+ })
138
+
139
+ describe('trySpend() - budget enforcement', () => {
140
+ it('should allow commands when under soft cap', () => {
141
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
142
+ try {
143
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
144
+ dailyLimit: 100,
145
+ reserveForCommands: 20,
146
+ })
147
+ // Use 50 requests (well under soft cap of 80)
148
+ for (let i = 0; i < 50; i++) {
149
+ expect(tracker.trySpend('command')).toBe(true)
150
+ }
151
+ expect(tracker.getCount()).toBe(50)
152
+ } finally {
153
+ cleanup(testDir)
154
+ }
155
+ })
156
+
157
+ it('should allow polling when under soft cap', () => {
158
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
159
+ try {
160
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
161
+ dailyLimit: 100,
162
+ reserveForCommands: 20,
163
+ })
164
+ // Use 50 requests
165
+ for (let i = 0; i < 50; i++) {
166
+ expect(tracker.trySpend('poll')).toBe(true)
167
+ }
168
+ expect(tracker.getCount()).toBe(50)
169
+ } finally {
170
+ cleanup(testDir)
171
+ }
172
+ })
173
+
174
+ it('should block polling at soft cap when pausePollingAtReserve is true', () => {
175
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
176
+ try {
177
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
178
+ dailyLimit: 100,
179
+ reserveForCommands: 20,
180
+ pausePollingAtReserve: true, // Enable soft cap blocking
181
+ })
182
+ // Use up to soft cap (80 requests)
183
+ for (let i = 0; i < 80; i++) {
184
+ tracker.track()
185
+ }
186
+ expect(tracker.getCount()).toBe(80)
187
+
188
+ // Polling should be blocked at soft cap
189
+ expect(tracker.trySpend('poll')).toBe(false)
190
+ expect(tracker.trySpend('discovery')).toBe(false)
191
+
192
+ // Commands should still work
193
+ expect(tracker.trySpend('command')).toBe(true)
194
+ expect(tracker.getCount()).toBe(81)
195
+ } finally {
196
+ cleanup(testDir)
197
+ }
198
+ })
199
+
200
+ it('should block all requests at hard cap', () => {
201
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
202
+ try {
203
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
204
+ dailyLimit: 100,
205
+ reserveForCommands: 20,
206
+ })
207
+ // Use up to hard cap
208
+ for (let i = 0; i < 100; i++) {
209
+ tracker.track()
210
+ }
211
+ expect(tracker.getCount()).toBe(100)
212
+
213
+ // All request types should be blocked
214
+ expect(tracker.trySpend('poll')).toBe(false)
215
+ expect(tracker.trySpend('discovery')).toBe(false)
216
+ expect(tracker.trySpend('command')).toBe(false)
217
+ expect(tracker.getCount()).toBe(100)
218
+ } finally {
219
+ cleanup(testDir)
220
+ }
221
+ })
222
+
223
+ it('should support batch spending', () => {
224
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
225
+ try {
226
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
227
+ dailyLimit: 100,
228
+ reserveForCommands: 20,
229
+ })
230
+ expect(tracker.trySpend('poll', 10)).toBe(true)
231
+ expect(tracker.getCount()).toBe(10)
232
+
233
+ expect(tracker.trySpend('command', 5)).toBe(true)
234
+ expect(tracker.getCount()).toBe(15)
235
+ } finally {
236
+ cleanup(testDir)
237
+ }
238
+ })
239
+ })
240
+
241
+ describe('webhookOnlyOnReserve mode', () => {
242
+ it('should continue polling beyond soft cap when pausePollingAtReserve is false', () => {
243
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
244
+ try {
245
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
246
+ dailyLimit: 100,
247
+ reserveForCommands: 20,
248
+ pausePollingAtReserve: false,
249
+ })
250
+ // Use 85 requests (past soft cap)
251
+ for (let i = 0; i < 85; i++) {
252
+ tracker.track()
253
+ }
254
+ expect(tracker.getCount()).toBe(85)
255
+
256
+ // Polling should still work (not paused at reserve)
257
+ expect(tracker.trySpend('poll')).toBe(true)
258
+ expect(tracker.getCount()).toBe(86)
259
+ } finally {
260
+ cleanup(testDir)
261
+ }
262
+ })
263
+
264
+ it('should stop polling at soft cap when pausePollingAtReserve is true', () => {
265
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
266
+ try {
267
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
268
+ dailyLimit: 100,
269
+ reserveForCommands: 20,
270
+ pausePollingAtReserve: true,
271
+ })
272
+ // Use up to soft cap
273
+ for (let i = 0; i < 80; i++) {
274
+ tracker.track()
275
+ }
276
+ expect(tracker.getCount()).toBe(80)
277
+
278
+ // Polling should be blocked
279
+ expect(tracker.trySpend('poll')).toBe(false)
280
+ expect(tracker.getCount()).toBe(80)
281
+
282
+ // Commands should still work
283
+ expect(tracker.trySpend('command')).toBe(true)
284
+ expect(tracker.getCount()).toBe(81)
285
+ } finally {
286
+ cleanup(testDir)
287
+ }
288
+ })
289
+ })
290
+
291
+ describe('warning logs', () => {
292
+ it('should log warning when reaching soft cap with pausePollingAtReserve enabled', () => {
293
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
294
+ try {
295
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
296
+ dailyLimit: 100,
297
+ reserveForCommands: 20,
298
+ pausePollingAtReserve: true, // Enable soft cap warning
299
+ })
300
+ // Use up to soft cap
301
+ for (let i = 0; i < 80; i++) {
302
+ tracker.track()
303
+ }
304
+
305
+ // Trigger soft cap warning by attempting poll (will be blocked)
306
+ tracker.trySpend('poll')
307
+ expect(mockLog.warn).toHaveBeenCalledWith(
308
+ expect.stringContaining('Near daily limit'),
309
+ )
310
+ } finally {
311
+ cleanup(testDir)
312
+ }
313
+ })
314
+
315
+ it('should log error when reaching hard cap', () => {
316
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
317
+ try {
318
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
319
+ dailyLimit: 100,
320
+ reserveForCommands: 20,
321
+ })
322
+ // Use up to hard cap
323
+ for (let i = 0; i < 100; i++) {
324
+ tracker.track()
325
+ }
326
+
327
+ // Trigger hard cap error
328
+ tracker.trySpend('command')
329
+ expect(mockLog.error).toHaveBeenCalledWith(
330
+ expect.stringContaining('Daily limit'),
331
+ )
332
+ } finally {
333
+ cleanup(testDir)
334
+ }
335
+ })
336
+ })
337
+
338
+ describe('hourly logging', () => {
339
+ it('should log immediately on startup', () => {
340
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
341
+ try {
342
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
343
+ tracker.startHourlyLogging()
344
+ tracker.stopHourlyLogging()
345
+ expect(mockLog.info).toHaveBeenCalledWith(
346
+ expect.stringContaining('[API Stats] Today'),
347
+ )
348
+ } finally {
349
+ cleanup(testDir)
350
+ }
351
+ })
352
+
353
+ it('should stop logging when stopHourlyLogging is called', () => {
354
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
355
+ try {
356
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest')
357
+ tracker.startHourlyLogging()
358
+ tracker.stopHourlyLogging()
359
+ // Should not throw
360
+ expect(true).toBe(true)
361
+ } finally {
362
+ cleanup(testDir)
363
+ }
364
+ })
365
+ })
366
+
367
+ describe('edge cases', () => {
368
+ it('should handle zero daily limit', () => {
369
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
370
+ try {
371
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
372
+ dailyLimit: 0,
373
+ reserveForCommands: 0,
374
+ })
375
+ // All requests should be blocked immediately
376
+ expect(tracker.trySpend('poll')).toBe(false)
377
+ expect(tracker.trySpend('command')).toBe(false)
378
+ expect(tracker.getCount()).toBe(0)
379
+ } finally {
380
+ cleanup(testDir)
381
+ }
382
+ })
383
+
384
+ it('should handle reserve larger than limit', () => {
385
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
386
+ try {
387
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
388
+ dailyLimit: 100,
389
+ reserveForCommands: 150,
390
+ pausePollingAtReserve: true, // Enable soft cap blocking
391
+ })
392
+ // Soft cap would be negative (100 - 150 = -50), clamped to 0
393
+ // With pausePollingAtReserve=true, polling should be blocked immediately
394
+ expect(tracker.trySpend('poll')).toBe(false)
395
+ // Commands up to hard cap should work
396
+ expect(tracker.trySpend('command')).toBe(true)
397
+ } finally {
398
+ cleanup(testDir)
399
+ }
400
+ })
401
+
402
+ it('should handle negative values in config', () => {
403
+ const { mockApi, mockLog, testDir } = createTestEnvironment()
404
+ try {
405
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
406
+ dailyLimit: -100,
407
+ reserveForCommands: -50,
408
+ })
409
+ // Should be treated as 0
410
+ expect(tracker.trySpend('poll')).toBe(false)
411
+ expect(tracker.trySpend('command')).toBe(false)
412
+ } finally {
413
+ cleanup(testDir)
414
+ }
415
+ })
416
+ })
417
+ })
@@ -0,0 +1,136 @@
1
+ /* eslint-disable import/first */
2
+ import { describe, expect, it, vi } from 'vitest'
3
+ // Mock modules used by HAP device base occasionally via platform
4
+ vi.mock('fakegato-history', () => ({ default: () => ({}) }))
5
+ vi.mock('homebridge-lib/EveHomeKitTypes', () => ({ EveHomeKitTypes: class { constructor() {} } }))
6
+
7
+ import { deviceBase } from '../../devices-hap/device.js'
8
+
9
+ // Minimal HAP stub to satisfy deviceBase constructor operations
10
+ function makeHapStub() {
11
+ class Service {
12
+ private _chars: Record<string, any> = {}
13
+ setCharacteristic(k: any, v: any) {
14
+ this._chars[k] = v
15
+ return this
16
+ }
17
+
18
+ getCharacteristic(k: any) {
19
+ return {
20
+ updateValue: (v: any) => {
21
+ this._chars[k] = v
22
+ return this
23
+ },
24
+ } as any
25
+ }
26
+ }
27
+ const Characteristic: any = {
28
+ Manufacturer: 'Manufacturer',
29
+ AppMatchingIdentifier: 'AppMatchingIdentifier',
30
+ Name: 'Name',
31
+ ConfiguredName: 'ConfiguredName',
32
+ Model: 'Model',
33
+ ProductData: 'ProductData',
34
+ SerialNumber: 'SerialNumber',
35
+ HardwareRevision: 'HardwareRevision',
36
+ SoftwareRevision: 'SoftwareRevision',
37
+ FirmwareRevision: 'FirmwareRevision',
38
+ On: 'On',
39
+ }
40
+ ;(Service as any).AccessoryInformation = class extends Service {}
41
+ ;(Service as any).Outlet = class extends Service {}
42
+ return { Service, Characteristic, Categories: { OUTLET: 7 } }
43
+ }
44
+
45
+ // Minimal PlatformAccessory stub
46
+ function makeAccessoryStub(hap: any, name = 'Test Accessory') {
47
+ const services: any[] = []
48
+ return {
49
+ displayName: name,
50
+ category: 0,
51
+ context: {},
52
+ getService(cls: any) {
53
+ // find existing or create
54
+ const svc = services.find(s => s instanceof cls)
55
+ if (svc) {
56
+ return svc
57
+ }
58
+ const s = new cls()
59
+ services.push(s)
60
+ return s
61
+ },
62
+ addService(cls: any) {
63
+ const s = new cls()
64
+ services.push(s)
65
+ return s
66
+ },
67
+ }
68
+ }
69
+
70
+ // Minimal SwitchBotHAPPlatform-like stub with required surface
71
+ function makePlatformStub(options: any, hap: any) {
72
+ const log = { info: vi.fn(), warn: vi.fn(), debug: vi.fn(), error: vi.fn(), success: vi.fn() }
73
+ return {
74
+ api: { hap },
75
+ log,
76
+ config: { options, credentials: {} },
77
+ debugMode: false,
78
+ // logging helpers used by deviceBase
79
+ infoLog: () => {},
80
+ successLog: () => {},
81
+ debugSuccessLog: () => {},
82
+ warnLog: () => {},
83
+ debugWarnLog: () => {},
84
+ errorLog: () => {},
85
+ debugErrorLog: () => {},
86
+ debugLog: () => {},
87
+ loggingIsDebug: async () => false,
88
+ enablingPlatformLogging: async () => true,
89
+ connectBLE: vi.fn(),
90
+ bleEventHandler: {},
91
+ webhookEventHandler: {},
92
+ }
93
+ }
94
+
95
+ // Create a tiny concrete subclass to instantiate deviceBase
96
+ class TestHAPDevice extends deviceBase {
97
+ // Override any methods that may be invoked by tests if needed
98
+ }
99
+
100
+ describe('hap device base webhook context', () => {
101
+ it('sets accessory.context.webhook=true when global webhook is enabled and device.webhook is undefined', async () => {
102
+ const hap = makeHapStub()
103
+ const accessory: any = makeAccessoryStub(hap, 'Plug Device')
104
+ const platform: any = makePlatformStub({ webhook: true, logging: 'debug' }, hap)
105
+
106
+ const dev: any = {
107
+ deviceId: 'DEV-HAP-1',
108
+ deviceType: 'Plug',
109
+ connectionType: 'OpenAPI',
110
+ // webhook intentionally undefined
111
+ }
112
+
113
+ const d = new TestHAPDevice(platform, accessory, dev)
114
+ // Assert context
115
+ expect(accessory.context.webhook).toBe(true)
116
+ // ensure no unused var warning
117
+ expect(d).toBeDefined()
118
+ })
119
+
120
+ it('keeps accessory.context.webhook=false when device.webhook=false even if global is true', async () => {
121
+ const hap = makeHapStub()
122
+ const accessory: any = makeAccessoryStub(hap, 'Plug Device 2')
123
+ const platform: any = makePlatformStub({ webhook: true, logging: 'debug' }, hap)
124
+
125
+ const dev: any = {
126
+ deviceId: 'DEV-HAP-2',
127
+ deviceType: 'Plug',
128
+ connectionType: 'OpenAPI',
129
+ webhook: false,
130
+ }
131
+
132
+ const d = new TestHAPDevice(platform, accessory, dev)
133
+ expect(accessory.context.webhook).toBe(false)
134
+ expect(d).toBeDefined()
135
+ })
136
+ })