@switchbot/homebridge-switchbot 5.0.0-beta.98 → 5.0.0

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 (282) hide show
  1. package/.changeset/config.json +14 -0
  2. package/.github/copilot-instructions.md +39 -0
  3. package/.github/workflows/ci.yml +4 -1
  4. package/.github/workflows/manual-e2e.yml +6 -3
  5. package/.github/workflows/release.yml +64 -15
  6. package/.github/workflows/stale.yml +2 -4
  7. package/.husky/pre-push +15 -0
  8. package/CHANGELOG.md +126 -134
  9. package/MIGRATION.md +16 -6
  10. package/README.md +84 -3
  11. package/TODO.md +263 -0
  12. package/config.schema.json +229 -36
  13. package/dist/SwitchBotHAPPlatform.d.ts +133 -0
  14. package/dist/SwitchBotHAPPlatform.d.ts.map +1 -0
  15. package/dist/SwitchBotHAPPlatform.js +555 -0
  16. package/dist/SwitchBotHAPPlatform.js.map +1 -0
  17. package/dist/SwitchBotMatterPlatform.d.ts +141 -0
  18. package/dist/SwitchBotMatterPlatform.d.ts.map +1 -0
  19. package/dist/SwitchBotMatterPlatform.js +536 -0
  20. package/dist/SwitchBotMatterPlatform.js.map +1 -0
  21. package/dist/device-types.d.ts +31 -0
  22. package/dist/device-types.d.ts.map +1 -0
  23. package/dist/device-types.js +246 -0
  24. package/dist/device-types.js.map +1 -0
  25. package/dist/deviceCommandMapper.d.ts +10 -0
  26. package/dist/deviceCommandMapper.d.ts.map +1 -0
  27. package/dist/deviceCommandMapper.js +319 -0
  28. package/dist/deviceCommandMapper.js.map +1 -0
  29. package/dist/deviceFactory.d.ts +3 -2
  30. package/dist/deviceFactory.d.ts.map +1 -1
  31. package/dist/deviceFactory.js +107 -29
  32. package/dist/deviceFactory.js.map +1 -1
  33. package/dist/devices/genericDevice.d.ts +59 -37
  34. package/dist/devices/genericDevice.d.ts.map +1 -1
  35. package/dist/devices/genericDevice.js +376 -78
  36. package/dist/devices/genericDevice.js.map +1 -1
  37. package/dist/errors.d.ts +38 -0
  38. package/dist/errors.d.ts.map +1 -0
  39. package/dist/errors.js +32 -0
  40. package/dist/errors.js.map +1 -0
  41. package/dist/homebridge-ui/device-types.js +246 -0
  42. package/dist/homebridge-ui/device-types.js.map +1 -0
  43. package/dist/homebridge-ui/deviceCommandMapper.js +319 -0
  44. package/dist/homebridge-ui/deviceCommandMapper.js.map +1 -0
  45. package/dist/homebridge-ui/endpoints/config.d.ts +3 -0
  46. package/dist/homebridge-ui/endpoints/config.d.ts.map +1 -0
  47. package/dist/homebridge-ui/endpoints/config.js +90 -0
  48. package/dist/homebridge-ui/endpoints/config.js.map +1 -0
  49. package/dist/homebridge-ui/endpoints/devices.d.ts +6 -0
  50. package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -0
  51. package/dist/homebridge-ui/endpoints/devices.js +144 -0
  52. package/dist/homebridge-ui/endpoints/devices.js.map +1 -0
  53. package/dist/homebridge-ui/endpoints/discovery.d.ts +7 -0
  54. package/dist/homebridge-ui/endpoints/discovery.d.ts.map +1 -0
  55. package/dist/homebridge-ui/endpoints/discovery.js +219 -0
  56. package/dist/homebridge-ui/endpoints/discovery.js.map +1 -0
  57. package/dist/homebridge-ui/errors.js +32 -0
  58. package/dist/homebridge-ui/errors.js.map +1 -0
  59. package/dist/homebridge-ui/homebridge-ui/endpoints/config.js +90 -0
  60. package/dist/homebridge-ui/homebridge-ui/endpoints/config.js.map +1 -0
  61. package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js +144 -0
  62. package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js.map +1 -0
  63. package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js +219 -0
  64. package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js.map +1 -0
  65. package/dist/homebridge-ui/homebridge-ui/server.js +11 -0
  66. package/dist/homebridge-ui/homebridge-ui/server.js.map +1 -0
  67. package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js +108 -0
  68. package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js.map +1 -0
  69. package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js +111 -0
  70. package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js.map +1 -0
  71. package/dist/homebridge-ui/homebridge-ui/utils/logger.js +17 -0
  72. package/dist/homebridge-ui/homebridge-ui/utils/logger.js.map +1 -0
  73. package/dist/homebridge-ui/public/css/styles.css +483 -0
  74. package/dist/homebridge-ui/public/index.html +197 -621
  75. package/dist/homebridge-ui/public/js/advanced-settings.d.ts +3 -0
  76. package/dist/homebridge-ui/public/js/advanced-settings.d.ts.map +1 -0
  77. package/dist/homebridge-ui/public/js/advanced-settings.js +95 -0
  78. package/dist/homebridge-ui/public/js/advanced-settings.js.map +1 -0
  79. package/dist/homebridge-ui/public/js/advanced-settings.ts +94 -0
  80. package/dist/homebridge-ui/public/js/api.d.ts +66 -0
  81. package/dist/homebridge-ui/public/js/api.d.ts.map +1 -0
  82. package/dist/homebridge-ui/public/js/api.js +295 -0
  83. package/dist/homebridge-ui/public/js/api.js.map +1 -0
  84. package/dist/homebridge-ui/public/js/api.ts +355 -0
  85. package/dist/homebridge-ui/public/js/app.d.ts +2 -0
  86. package/dist/homebridge-ui/public/js/app.d.ts.map +1 -0
  87. package/dist/homebridge-ui/public/js/app.js +3722 -0
  88. package/dist/homebridge-ui/public/js/app.js.map +7 -0
  89. package/dist/homebridge-ui/public/js/app.ts +22 -0
  90. package/dist/homebridge-ui/public/js/constants.d.ts +2 -0
  91. package/dist/homebridge-ui/public/js/constants.d.ts.map +1 -0
  92. package/dist/homebridge-ui/public/js/constants.js +2 -0
  93. package/dist/homebridge-ui/public/js/constants.js.map +1 -0
  94. package/dist/homebridge-ui/public/js/constants.ts +1 -0
  95. package/dist/homebridge-ui/public/js/credentials.d.ts +3 -0
  96. package/dist/homebridge-ui/public/js/credentials.d.ts.map +1 -0
  97. package/dist/homebridge-ui/public/js/credentials.js +99 -0
  98. package/dist/homebridge-ui/public/js/credentials.js.map +1 -0
  99. package/dist/homebridge-ui/public/js/credentials.ts +105 -0
  100. package/dist/homebridge-ui/public/js/devices-delete.d.ts +3 -0
  101. package/dist/homebridge-ui/public/js/devices-delete.d.ts.map +1 -0
  102. package/dist/homebridge-ui/public/js/devices-delete.js +199 -0
  103. package/dist/homebridge-ui/public/js/devices-delete.js.map +1 -0
  104. package/dist/homebridge-ui/public/js/devices-delete.ts +227 -0
  105. package/dist/homebridge-ui/public/js/devices.d.ts +9 -0
  106. package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -0
  107. package/dist/homebridge-ui/public/js/devices.js +98 -0
  108. package/dist/homebridge-ui/public/js/devices.js.map +1 -0
  109. package/dist/homebridge-ui/public/js/devices.ts +106 -0
  110. package/dist/homebridge-ui/public/js/discovery.d.ts +9 -0
  111. package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -0
  112. package/dist/homebridge-ui/public/js/discovery.js +1201 -0
  113. package/dist/homebridge-ui/public/js/discovery.js.map +1 -0
  114. package/dist/homebridge-ui/public/js/discovery.ts +1335 -0
  115. package/dist/homebridge-ui/public/js/logger.d.ts +7 -0
  116. package/dist/homebridge-ui/public/js/logger.d.ts.map +1 -0
  117. package/dist/homebridge-ui/public/js/logger.js +17 -0
  118. package/dist/homebridge-ui/public/js/logger.js.map +1 -0
  119. package/dist/homebridge-ui/public/js/logger.ts +17 -0
  120. package/dist/homebridge-ui/public/js/modal.d.ts +5 -0
  121. package/dist/homebridge-ui/public/js/modal.d.ts.map +1 -0
  122. package/dist/homebridge-ui/public/js/modal.js +35 -0
  123. package/dist/homebridge-ui/public/js/modal.js.map +1 -0
  124. package/dist/homebridge-ui/public/js/modal.ts +35 -0
  125. package/dist/homebridge-ui/public/js/modals.d.ts +15 -0
  126. package/dist/homebridge-ui/public/js/modals.d.ts.map +1 -0
  127. package/dist/homebridge-ui/public/js/modals.js +675 -0
  128. package/dist/homebridge-ui/public/js/modals.js.map +1 -0
  129. package/dist/homebridge-ui/public/js/modals.ts +765 -0
  130. package/dist/homebridge-ui/public/js/render.d.ts +71 -0
  131. package/dist/homebridge-ui/public/js/render.d.ts.map +1 -0
  132. package/dist/homebridge-ui/public/js/render.js +960 -0
  133. package/dist/homebridge-ui/public/js/render.js.map +1 -0
  134. package/dist/homebridge-ui/public/js/render.ts +1084 -0
  135. package/dist/homebridge-ui/public/js/toast.d.ts +6 -0
  136. package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -0
  137. package/dist/homebridge-ui/public/js/toast.js +38 -0
  138. package/dist/homebridge-ui/public/js/toast.js.map +1 -0
  139. package/dist/homebridge-ui/public/js/toast.ts +44 -0
  140. package/dist/homebridge-ui/public/js/types.d.ts +23 -0
  141. package/dist/homebridge-ui/public/js/types.d.ts.map +1 -0
  142. package/dist/homebridge-ui/public/js/types.js +2 -0
  143. package/dist/homebridge-ui/public/js/types.js.map +1 -0
  144. package/dist/homebridge-ui/public/js/types.ts +26 -0
  145. package/dist/homebridge-ui/server.d.ts +1 -3
  146. package/dist/homebridge-ui/server.d.ts.map +1 -1
  147. package/dist/homebridge-ui/server.js +8 -450
  148. package/dist/homebridge-ui/server.js.map +1 -1
  149. package/dist/homebridge-ui/settings.js +8 -0
  150. package/dist/homebridge-ui/settings.js.map +1 -0
  151. package/dist/homebridge-ui/switchbotClient.js +247 -0
  152. package/dist/homebridge-ui/switchbotClient.js.map +1 -0
  153. package/dist/homebridge-ui/utils/config-parser.d.ts +39 -0
  154. package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -0
  155. package/dist/homebridge-ui/utils/config-parser.js +108 -0
  156. package/dist/homebridge-ui/utils/config-parser.js.map +1 -0
  157. package/dist/homebridge-ui/utils/device-migration.d.ts +35 -0
  158. package/dist/homebridge-ui/utils/device-migration.d.ts.map +1 -0
  159. package/dist/homebridge-ui/utils/device-migration.js +111 -0
  160. package/dist/homebridge-ui/utils/device-migration.js.map +1 -0
  161. package/dist/homebridge-ui/utils/logger.d.ts +7 -0
  162. package/dist/homebridge-ui/utils/logger.d.ts.map +1 -0
  163. package/dist/homebridge-ui/utils/logger.js +17 -0
  164. package/dist/homebridge-ui/utils/logger.js.map +1 -0
  165. package/dist/index.d.ts +10 -0
  166. package/dist/index.d.ts.map +1 -1
  167. package/dist/index.js +12 -2
  168. package/dist/index.js.map +1 -1
  169. package/dist/settings.d.ts +1 -0
  170. package/dist/settings.d.ts.map +1 -1
  171. package/dist/settings.js +1 -0
  172. package/dist/settings.js.map +1 -1
  173. package/dist/switchbotClient.d.ts +12 -10
  174. package/dist/switchbotClient.d.ts.map +1 -1
  175. package/dist/switchbotClient.js +156 -103
  176. package/dist/switchbotClient.js.map +1 -1
  177. package/dist/utils.d.ts +76 -1
  178. package/dist/utils.d.ts.map +1 -1
  179. package/dist/utils.js +1121 -4
  180. package/dist/utils.js.map +1 -1
  181. package/docs/assets/highlight.css +16 -2
  182. package/docs/assets/main.js +1 -1
  183. package/docs/index.html +82 -5
  184. package/docs/variables/default.html +3 -1
  185. package/eslint.config.js +9 -5
  186. package/nodemon.json +2 -2
  187. package/package.json +34 -21
  188. package/scripts/build-ui.js +37 -0
  189. package/scripts/free-dev-ports.mjs +105 -0
  190. package/scripts/generate-matter-maps.js +34 -17
  191. package/scripts/sync-device-types.mjs +31 -0
  192. package/src/SwitchBotHAPPlatform.ts +558 -0
  193. package/src/SwitchBotMatterPlatform.ts +538 -0
  194. package/src/device-types.js +246 -0
  195. package/src/device-types.js.map +1 -0
  196. package/src/device-types.ts +261 -0
  197. package/src/deviceCommandMapper.js +319 -0
  198. package/src/deviceCommandMapper.js.map +1 -0
  199. package/src/deviceCommandMapper.ts +333 -0
  200. package/src/deviceFactory.ts +125 -45
  201. package/src/devices/genericDevice.ts +411 -69
  202. package/src/errors.js +32 -0
  203. package/src/errors.js.map +1 -0
  204. package/src/errors.ts +35 -0
  205. package/src/homebridge-ui/endpoints/config.ts +110 -0
  206. package/src/homebridge-ui/endpoints/devices.ts +153 -0
  207. package/src/homebridge-ui/endpoints/discovery.ts +240 -0
  208. package/src/homebridge-ui/public/css/styles.css +483 -0
  209. package/src/homebridge-ui/public/index.html +197 -621
  210. package/src/homebridge-ui/public/js/advanced-settings.ts +94 -0
  211. package/src/homebridge-ui/public/js/api.ts +355 -0
  212. package/src/homebridge-ui/public/js/app.ts +22 -0
  213. package/src/homebridge-ui/public/js/constants.ts +1 -0
  214. package/src/homebridge-ui/public/js/credentials.ts +105 -0
  215. package/src/homebridge-ui/public/js/devices-delete.ts +227 -0
  216. package/src/homebridge-ui/public/js/devices.ts +106 -0
  217. package/src/homebridge-ui/public/js/discovery.ts +1335 -0
  218. package/src/homebridge-ui/public/js/logger.ts +17 -0
  219. package/src/homebridge-ui/public/js/modal.ts +35 -0
  220. package/src/homebridge-ui/public/js/modals.ts +765 -0
  221. package/src/homebridge-ui/public/js/render.ts +1084 -0
  222. package/src/homebridge-ui/public/js/toast.ts +44 -0
  223. package/src/homebridge-ui/public/js/types.ts +26 -0
  224. package/src/homebridge-ui/server.ts +9 -526
  225. package/src/homebridge-ui/utils/config-parser.ts +125 -0
  226. package/src/homebridge-ui/utils/device-migration.ts +144 -0
  227. package/src/homebridge-ui/utils/logger.ts +17 -0
  228. package/src/index.ts +12 -2
  229. package/src/settings.js +8 -0
  230. package/src/settings.js.map +1 -0
  231. package/src/settings.ts +2 -0
  232. package/src/switchbotClient.js +247 -0
  233. package/src/switchbotClient.js.map +1 -0
  234. package/src/switchbotClient.ts +177 -114
  235. package/src/utils.ts +1133 -5
  236. package/test/client/switchbot-client-debounce.spec.ts +35 -0
  237. package/test/client/switchbot-client-openapi.spec.ts +19 -0
  238. package/test/client/switchbotClient.spec.ts +64 -0
  239. package/test/device/device-mapping.spec.ts +23 -0
  240. package/test/device/deviceBase.spec.ts +26 -0
  241. package/test/device/deviceFactory-edge.spec.ts +15 -0
  242. package/test/device/deviceFactory.spec.ts +33 -0
  243. package/test/device/fan-swing.spec.ts +34 -0
  244. package/test/device/genericDevice-blepoll.spec.ts +47 -0
  245. package/test/device/irdevice.spec.ts +9 -0
  246. package/test/device/lock-users.spec.ts +35 -0
  247. package/test/device/matter-descriptors.spec.ts +22 -0
  248. package/test/device/matter-device-state.spec.ts +37 -0
  249. package/test/e2e/run-e2e.spec.ts +18 -19
  250. package/test/errors/errors.spec.ts +10 -0
  251. package/test/helpers/matter-harness.ts +20 -9
  252. package/test/homebridge-ui/server.spec.ts +9 -0
  253. package/test/platform/accessory-restore.spec.ts +37 -0
  254. package/test/platform/matter-childbridge.spec.ts +34 -0
  255. package/test/platform/matter-integration.spec.ts +33 -0
  256. package/test/platform/platform-edge.spec.ts +73 -0
  257. package/test/platform/platform.integration.spec.ts +34 -0
  258. package/test/utils/utils-extra.spec.ts +10 -0
  259. package/test/utils/utils.spec.ts +53 -0
  260. package/todo/TODO.md +80 -0
  261. package/tsconfig.ui.json +11 -0
  262. package/.github/npm-version-script-esm.js +0 -97
  263. package/.github/workflows/beta-release.yml +0 -52
  264. package/dist/platform.d.ts +0 -35
  265. package/dist/platform.d.ts.map +0 -1
  266. package/dist/platform.js +0 -850
  267. package/dist/platform.js.map +0 -1
  268. package/src/platform.ts +0 -867
  269. package/test/accessory-restore.spec.ts +0 -73
  270. package/test/device-mapping.spec.ts +0 -37
  271. package/test/deviceFactory.spec.ts +0 -18
  272. package/test/fan-swing.spec.ts +0 -29
  273. package/test/lock-users.spec.ts +0 -44
  274. package/test/matter-childbridge.spec.ts +0 -55
  275. package/test/matter-descriptors.spec.ts +0 -97
  276. package/test/matter-device-state.spec.ts +0 -101
  277. package/test/matter-integration.spec.ts +0 -70
  278. package/test/platform.integration.spec.ts +0 -55
  279. package/test/switchbot-client-debounce.spec.ts +0 -131
  280. package/test/switchbot-client-openapi.spec.ts +0 -56
  281. package/test/switchbotClient.spec.ts +0 -10
  282. package/test/utils.spec.ts +0 -20
@@ -0,0 +1,73 @@
1
+ import { describe, expect, it } from 'vitest'
2
+
3
+ import SwitchBotHAPPlatform from '../../src/SwitchBotHAPPlatform.js'
4
+ import { SwitchBotMatterPlatform } from '../../src/SwitchBotMatterPlatform.js'
5
+
6
+ const mockLogger = {
7
+ info: () => {},
8
+ warn: () => {},
9
+ error: () => {},
10
+ debug: () => {},
11
+ success: () => {},
12
+ log: () => {},
13
+ }
14
+ const mockApi = { isMatterAvailable: () => false, isMatterEnabled: () => false } as any
15
+
16
+ describe('platform edge cases', () => {
17
+ const platforms = [
18
+ { name: 'SwitchBotHAPPlatform', Platform: SwitchBotHAPPlatform },
19
+ { name: 'SwitchBotMatterPlatform', Platform: SwitchBotMatterPlatform },
20
+ ]
21
+
22
+ for (const { name, Platform } of platforms) {
23
+ describe(`${name}`, () => {
24
+ it('should handle missing config', () => {
25
+ expect(() => new Platform(mockLogger, { platform: 'SwitchBot' }, mockApi as any)).not.toThrow()
26
+ })
27
+
28
+ it('should handle undefined config', () => {
29
+ expect(() => new Platform(mockLogger, undefined as any, mockApi)).not.toThrow()
30
+ })
31
+
32
+ it('should handle null config', () => {
33
+ expect(() => new Platform(mockLogger, null as any, mockApi)).not.toThrow()
34
+ })
35
+
36
+ it('should handle missing api', () => {
37
+ expect(() => new Platform(mockLogger, { platform: 'SwitchBot' }, undefined as any)).not.toThrow()
38
+ })
39
+
40
+ it('should handle null api', () => {
41
+ expect(() => new Platform(mockLogger, { platform: 'SwitchBot' }, null as any)).not.toThrow()
42
+ })
43
+
44
+ it('should handle missing logger', () => {
45
+ expect(() => new Platform(undefined as any, { platform: 'SwitchBot' }, mockApi)).toThrow()
46
+ })
47
+
48
+ it('should handle null logger', () => {
49
+ expect(() => new Platform(null as any, { platform: 'SwitchBot' }, mockApi)).toThrow()
50
+ })
51
+
52
+ it('should not throw with minimal valid config', () => {
53
+ const config = { name: 'SwitchBot', platform: 'SwitchBot' }
54
+ expect(() => new Platform(mockLogger, config, mockApi)).not.toThrow()
55
+ })
56
+
57
+ it('should fallback gracefully if isMatterAvailable returns true but isMatterEnabled returns false', () => {
58
+ const api = { ...mockApi, isMatterAvailable: () => true, isMatterEnabled: () => false }
59
+ expect(() => new Platform(mockLogger, { platform: 'SwitchBot' }, api)).not.toThrow()
60
+ })
61
+
62
+ it('should fallback gracefully if isMatterAvailable returns false', () => {
63
+ const api = { ...mockApi, isMatterAvailable: () => false }
64
+ expect(() => new Platform(mockLogger, { platform: 'SwitchBot' }, api)).not.toThrow()
65
+ })
66
+
67
+ it('should not throw if config has extra unknown properties', () => {
68
+ const config = { name: 'SwitchBot', platform: 'SwitchBot', extra: 123 }
69
+ expect(() => new Platform(mockLogger, config, mockApi)).not.toThrow()
70
+ })
71
+ })
72
+ }
73
+ })
@@ -0,0 +1,34 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import { SwitchBotHAPPlatform } from '../../src/SwitchBotHAPPlatform.js'
4
+
5
+ describe('hAP platform integration', () => {
6
+ class TestHAPPlatform extends SwitchBotHAPPlatform {
7
+ constructor(logger: any, config: any, api: any) { super(logger, config, api) }
8
+ setCache(cache: any[]) { (this as any)._accessoryCache = cache }
9
+ getRestored() { return (this as any)._accessoryCache }
10
+ registerAccessory(acc: any) { (this as any)._accessoryCache.push(acc) }
11
+ }
12
+
13
+ const logger = { info: vi.fn(), warn: vi.fn(), error: vi.fn(), debug: vi.fn() }
14
+ const config = {}
15
+ const api = {}
16
+
17
+ it('should restore accessories from cache', () => {
18
+ const platform = new TestHAPPlatform(logger, config, api)
19
+ platform.setCache([{ uuid: 'hap-acc-1', context: { type: 'Bot' } }])
20
+ const restored = platform.getRestored()
21
+ expect(restored).toHaveLength(1)
22
+ expect(restored[0].uuid).toBe('hap-acc-1')
23
+ })
24
+
25
+ it('should register a new accessory', () => {
26
+ const platform = new TestHAPPlatform(logger, config, api)
27
+ platform.setCache([])
28
+ const newAcc = { uuid: 'hap-acc-2', context: { type: 'Curtain' } }
29
+ platform.registerAccessory(newAcc)
30
+ const restored = platform.getRestored()
31
+ expect(restored).toHaveLength(1)
32
+ expect(restored[0].uuid).toBe('hap-acc-2')
33
+ })
34
+ })
@@ -0,0 +1,10 @@
1
+ import { describe, expect, it } from 'vitest'
2
+
3
+ import * as utils from '../../src/utils'
4
+
5
+ describe('extra utils', () => {
6
+ it('normalizeConfig returns empty object for undefined', () => {
7
+ expect(utils.normalizeConfig(undefined)).toEqual({})
8
+ })
9
+ // Add more utility tests as needed
10
+ })
@@ -0,0 +1,53 @@
1
+ import { describe, expect, it, vi } from 'vitest'
2
+
3
+ import {
4
+ createPlatformProxy,
5
+ MATTER_ATTRIBUTE_IDS,
6
+ MATTER_CLUSTER_IDS,
7
+ normalizeConfig,
8
+ } from '../../src/utils'
9
+
10
+ describe('utils', () => {
11
+ it('should expose correct MATTER_CLUSTER_IDS', () => {
12
+ expect(MATTER_CLUSTER_IDS.OnOff).toBe(0x0006)
13
+ expect(MATTER_CLUSTER_IDS.FanControl).toBe(0x0202)
14
+ })
15
+
16
+ it('should expose correct MATTER_ATTRIBUTE_IDS', () => {
17
+ expect(MATTER_ATTRIBUTE_IDS.OnOff.OnOff).toBe(0x0000)
18
+ expect(MATTER_ATTRIBUTE_IDS.ColorControl.CurrentHue).toBe(0x0000)
19
+ })
20
+
21
+ it('normalizeConfig returns empty object for undefined', () => {
22
+ expect(normalizeConfig(undefined)).toEqual({})
23
+ })
24
+
25
+ it('normalizeConfig returns shallow copy of config', () => {
26
+ const input = { foo: 'bar', preferMatter: false }
27
+ const result = normalizeConfig(input as any)
28
+ expect(result).toMatchObject(input)
29
+ expect(result).not.toBe(input)
30
+ })
31
+
32
+ it('createPlatformProxy instantiates MatterPlatform if available and enabled', () => {
33
+ const HAP = vi.fn()
34
+ const Matter = vi.fn()
35
+ const api = { isMatterAvailable: () => true, isMatterEnabled: () => true }
36
+ const config = { enableMatter: true, preferMatter: true }
37
+ const Proxy = createPlatformProxy(HAP, Matter)
38
+ new Proxy('log', config, api)
39
+ expect(Matter).toHaveBeenCalled()
40
+ expect(HAP).not.toHaveBeenCalled()
41
+ })
42
+
43
+ it('createPlatformProxy falls back to HAPPlatform if Matter not available', () => {
44
+ const HAP = vi.fn()
45
+ const Matter = vi.fn()
46
+ const api = { isMatterAvailable: () => false, isMatterEnabled: () => false }
47
+ const config = { enableMatter: true, preferMatter: true }
48
+ const Proxy = createPlatformProxy(HAP, Matter)
49
+ new Proxy('log', config, api)
50
+ expect(HAP).toHaveBeenCalled()
51
+ expect(Matter).not.toHaveBeenCalled()
52
+ })
53
+ })
package/todo/TODO.md ADDED
@@ -0,0 +1,80 @@
1
+ # SwitchBot Plugin TODO List
2
+
3
+ - [ ] Implement battery level trending (if historical data available)
4
+ - [ ] Add "Export Results" button (JSON format)
5
+ - [ ] Add "Import Results" for comparison or troubleshooting
6
+ - [ ] Share with support for debugging
7
+ - [ ] Compare current vs previous discovery (show differences)
8
+ - [ ] Generate discovery report (formatted summary)
9
+ - [ ] **Test discovery with real hardware**
10
+ - [ ] Test BLE discovery with nearby devices
11
+ - [ ] Test OpenAPI discovery with cloud-registered devices
12
+ - [ ] Test with mixed BLE+API devices
13
+ - [ ] Test connection type badge display
14
+ - [ ] Verify deduplication works correctly
15
+ - [ ] **Add error handling improvements**
16
+ - [ ] Better error messages for users
17
+ - [ ] Retry logic with exponential backoff
18
+ - [ ] Graceful degradation when BLE unavailable
19
+ - [ ] Network timeout handling
20
+ - [ ] **Update README with discovery features**
21
+ - [ ] Document BLE vs OpenAPI discovery
22
+ - [ ] Add screenshots of new UI features
23
+ - [ ] Explain connection type badges
24
+ - [ ] Add troubleshooting guide for discovery issues
25
+ - [ ] **Create developer documentation**
26
+ - [ ] Document new modular structure
27
+ - [ ] API documentation for endpoints
28
+ - [ ] Contributing guidelines for UI changes
29
+ - [ ] Better Loading States
30
+ - [ ] Connection Recommendations
31
+ - [ ] Device Grouping
32
+ - [ ] Test discovery with real hardware
33
+ - [ ] Update README with new features
34
+ - [ ] Configurable BLE Scan
35
+ - [ ] Discovery History/Cache
36
+ - [ ] Connection Testing
37
+ - [ ] Batch Import
38
+ - [ ] Device Health Dashboard
39
+ - [ ] Add checkboxes to select multiple devices
40
+ - [ ] Add "Add Selected to Config" button
41
+ - [ ] Implement bulk configuration with smart defaults
42
+ - [ ] Add "Needs Attention" section at top
43
+ - [ ] Implement battery level trending (if historical data available)
44
+ - [ ] Add "Export Results" button (JSON format)
45
+ - [ ] Add "Import Results" for comparison or troubleshooting
46
+ - [ ] Share with support for debugging
47
+ - [ ] Compare current vs previous discovery (show differences)
48
+ - [ ] Generate discovery report (formatted summary)
49
+ - [ ] Add tests for battery level trending logic
50
+ - [ ] **Test discovery with real hardware**
51
+ - [ ] Add tests for battery level trending logic
52
+ - [ ] Test BLE discovery with nearby devices
53
+ - [ ] Test OpenAPI discovery with cloud-registered devices
54
+ - [ ] Test with mixed BLE+API devices
55
+ - [ ] Test connection type badge display
56
+ - [ ] Verify deduplication works correctly
57
+ - [ ] **Add error handling improvements**
58
+ - [ ] Better error messages for users
59
+ - [ ] Retry logic with exponential backoff
60
+ - [ ] Graceful degradation when BLE unavailable
61
+ - [ ] Network timeout handling
62
+ - [ ] **Update README with discovery features**
63
+ - [ ] Document BLE vs OpenAPI discovery
64
+ - [ ] Add screenshots of new UI features
65
+ - [ ] Explain connection type badges
66
+ - [ ] Add troubleshooting guide for discovery issues
67
+ - [ ] **Create developer documentation**
68
+ - [ ] Document new modular structure
69
+ - [ ] API documentation for endpoints
70
+ - [ ] Contributing guidelines for UI changes
71
+ - [ ] Better Loading States
72
+ - [ ] Connection Recommendations
73
+ - [ ] Device Grouping
74
+ - [ ] Test discovery with real hardware
75
+ - [ ] Update README with new features
76
+ - [ ] Configurable BLE Scan
77
+ - [ ] Discovery History/Cache
78
+ - [ ] Connection Testing
79
+ - [ ] Batch Import
80
+ - [ ] Device Health Dashboard
@@ -0,0 +1,11 @@
1
+ {
2
+ "extends": "./tsconfig.json",
3
+ "compilerOptions": {
4
+ "declaration": false,
5
+ "declarationMap": false,
6
+ "outDir": "dist/homebridge-ui"
7
+ },
8
+ "include": [
9
+ "src/homebridge-ui/server.ts"
10
+ ]
11
+ }
@@ -1,97 +0,0 @@
1
- #!/bin/env node
2
-
3
- /**
4
- * This scripts queries the npm registry to pull out the latest version for a given tag.
5
- */
6
-
7
- import assert from 'node:assert'
8
- import child_process from 'node:child_process'
9
- import fs from 'node:fs'
10
- import process from 'node:process'
11
-
12
- const BRANCH_VERSION_PATTERN = /^([A-Z]+)-(\d+\.\d+\.\d+)$/i
13
-
14
- // Load the contents of the package.json file
15
- const packageJSON = JSON.parse(fs.readFileSync('package.json', 'utf8'))
16
-
17
- const refArgument = process.argv[2]
18
- const tagArgument = process.argv[3] || 'latest'
19
-
20
- if (!refArgument) {
21
- console.error('ref argument is missing')
22
- console.error('Usage: npm-version-script-esm.js <ref> [tag]')
23
- process.exit(1)
24
- }
25
-
26
- /**
27
- * Queries the NPM registry for the latest version for the provided base version and tag.
28
- * If the tag is latest, then the base version is returned if it exists. For other tags, the latest
29
- * version found for that base version and tag is returned.
30
- * @param baseVersion The base version to query for, e.g. 2.0.0
31
- * @param tag The tag to query for, e.g. beta or latest
32
- * @returns {string} Returns the version, or '' if not found
33
- */
34
- function getTagVersionFromNpm(baseVersion, tag) {
35
- try {
36
- return JSON.parse(child_process.execSync(`npm info ${packageJSON.name} versions --json`).toString('utf8').trim())
37
- .filter(v => tag === 'latest' ? v === baseVersion : v.startsWith(`${baseVersion}-${tag}.`)) // find all published versions for this base version and tag
38
- .reduce((_, current) => current, '') // choose the last as they're sorted in ascending order, or '' if there are none
39
- } catch (e) {
40
- console.error(`Failed to query the npm registry for the latest version for tag: ${tag}`, e)
41
- // throw e;
42
- return ''
43
- }
44
- }
45
-
46
- function desiredTargetVersion(ref) {
47
- // ref is a GitHub action ref string
48
- if (ref.startsWith('refs/pull/')) {
49
- throw new Error('The version script was executed inside a PR!')
50
- }
51
-
52
- assert(ref.startsWith('refs/heads/'))
53
- const branchName = ref.slice('refs/heads/'.length)
54
-
55
- const results = branchName.match(BRANCH_VERSION_PATTERN)
56
- if (results !== null) {
57
- if (results[1] !== tagArgument) {
58
- console.warn(`The base branch name (${results[1]}) differs from the tag name ${tagArgument}`)
59
- }
60
-
61
- return results[2]
62
- }
63
-
64
- throw new Error(`Malformed branch name for ref: ${ref}. Can't derive the base version. Use a branch name like: beta-x.x.x or alpha-x.x.x`)
65
- }
66
-
67
- // derive the base version from the branch ref
68
- const baseVersion = desiredTargetVersion(refArgument)
69
-
70
- // query the npm registry for the latest version of the provided tag name
71
- const latestReleasedVersion = getTagVersionFromNpm(baseVersion, tagArgument) // e.g. 0.7.0-beta.12
72
-
73
- let publishTag
74
-
75
- if (latestReleasedVersion) {
76
- console.warn(`Latest published version for ${baseVersion} with tag ${tagArgument} is ${latestReleasedVersion}`)
77
- publishTag = latestReleasedVersion // set this released beta or alpha to be incremented
78
- } else {
79
- console.warn(`No published versions for ${baseVersion} with tag ${tagArgument}`)
80
- publishTag = baseVersion // start off with a new beta or alpha version
81
- }
82
-
83
- if (packageJSON.version !== publishTag) {
84
- // report the change for CI
85
- console.warn(`Changing version in package.json from ${packageJSON.version} to ${publishTag}`)
86
-
87
- // save the package.json
88
- packageJSON.version = publishTag
89
- fs.writeFileSync('package.json', JSON.stringify(packageJSON, null, 2))
90
-
91
- // perform the same change to the package-lock.json
92
- const packageLockJSON = JSON.parse(fs.readFileSync('package-lock.json', 'utf8'))
93
- packageLockJSON.version = publishTag
94
- fs.writeFileSync('package-lock.json', JSON.stringify(packageLockJSON, null, 2))
95
- } else {
96
- console.warn(`Leaving version in package.json at ${packageJSON.version}`)
97
- }
@@ -1,52 +0,0 @@
1
- name: Beta Release
2
-
3
- on:
4
- push:
5
- branches: [beta-*.*.*, beta]
6
- workflow_dispatch:
7
-
8
- jobs:
9
- build_and_test:
10
- uses: homebridge/.github/.github/workflows/nodejs-build-and-test.yml@latest
11
- with:
12
- enable_coverage: false
13
- secrets:
14
- token: ${{ secrets.GITHUB_TOKEN }}
15
- lint:
16
- needs: build_and_test
17
- uses: homebridge/.github/.github/workflows/eslint.yml@latest
18
-
19
- publish:
20
- needs: lint
21
- permissions:
22
- id-token: write
23
- uses: homebridge/.github/.github/workflows/npm-publish-esm.yml@latest
24
- with:
25
- tag: 'beta'
26
- dynamically_adjust_version: true
27
- npm_version_command: 'pre'
28
- pre_id: 'beta'
29
- secrets:
30
- npm_auth_token: ${{ secrets.npm_token }}
31
-
32
- pre-release:
33
- needs: publish
34
- uses: homebridge/.github/.github/workflows/pre-release.yml@latest
35
- with:
36
- npm_version: ${{ needs.publish.outputs.NPM_VERSION }}
37
- body: |
38
- **Beta Release**
39
- **Version**: v${{ needs.publish.outputs.NPM_VERSION }}
40
- [How To Test Beta Releases](https://github.com/OpenWonderLabs/homebridge-switchbot/wiki/Beta-Version)
41
-
42
- github-releases-to-discord:
43
- name: Discord Webhooks
44
- needs: [build_and_test,publish]
45
- uses: homebridge/.github/.github/workflows/discord-webhooks.yml@latest
46
- with:
47
- title: "SwitchBot Beta Release"
48
- description: |
49
- Version `v${{ needs.publish.outputs.NPM_VERSION }}`
50
- url: "https://github.com/OpenWonderLabs/homebridge-switchbot/releases/tag/v${{ needs.publish.outputs.NPM_VERSION }}"
51
- secrets:
52
- DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK_URL_LATEST }}
@@ -1,35 +0,0 @@
1
- import type { SwitchBotPluginConfig } from './settings.js';
2
- import type { API, Logger, PlatformConfig } from 'homebridge';
3
- export declare class SwitchBotHAPPlatform {
4
- api: API | undefined;
5
- log: Logger;
6
- config: SwitchBotPluginConfig;
7
- devices: any[];
8
- accessories: Map<string, any>;
9
- private lastConfigHash;
10
- private configReloadInterval;
11
- constructor(log: Logger, config: PlatformConfig, api?: API);
12
- private getConfigHash;
13
- private checkAndReloadDevices;
14
- loadDevices(): Promise<void>;
15
- configureAccessory(accessory: any): Promise<void>;
16
- configureMatterAccessory?(accessory: any): void;
17
- }
18
- export declare class SwitchBotMatterPlatform {
19
- api: API | undefined;
20
- log: Logger;
21
- config: SwitchBotPluginConfig;
22
- devices: any[];
23
- accessories: Map<string, any>;
24
- private lastConfigHash;
25
- private configReloadInterval;
26
- constructor(log: Logger, config: PlatformConfig, api?: API);
27
- loadDevices(): Promise<void>;
28
- private getConfigHash;
29
- private checkAndReloadDevices;
30
- configureAccessory(accessory: any): Promise<void>;
31
- configureMatterAccessory(accessory: any): void;
32
- registerMatterAccessories(): Promise<void>;
33
- }
34
- export default SwitchBotHAPPlatform;
35
- //# sourceMappingURL=platform.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"platform.d.ts","sourceRoot":"","sources":["../src/platform.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAA;AAC1D,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,YAAY,CAAA;AA0P7D,qBAAa,oBAAoB;IAC/B,GAAG,EAAE,GAAG,GAAG,SAAS,CAAA;IACpB,GAAG,EAAE,MAAM,CAAA;IACX,MAAM,EAAE,qBAAqB,CAAA;IAC7B,OAAO,EAAE,GAAG,EAAE,CAAK;IAEnB,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;IAE7B,OAAO,CAAC,cAAc,CAAa;IACnC,OAAO,CAAC,oBAAoB,CAA8B;gBAE9C,GAAG,EAAE,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,GAAG,CAAC,EAAE,GAAG;IA2C1D,OAAO,CAAC,aAAa;YAUP,qBAAqB;IAU7B,WAAW;IAoMX,kBAAkB,CAAC,SAAS,EAAE,GAAG;IAYvC,wBAAwB,CAAC,CAAC,SAAS,EAAE,GAAG;CASzC;AAGD,qBAAa,uBAAuB;IAClC,GAAG,EAAE,GAAG,GAAG,SAAS,CAAA;IACpB,GAAG,EAAE,MAAM,CAAA;IACX,MAAM,EAAE,qBAAqB,CAAA;IAC7B,OAAO,EAAE,GAAG,EAAE,CAAK;IACnB,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;IAE7B,OAAO,CAAC,cAAc,CAAa;IACnC,OAAO,CAAC,oBAAoB,CAA8B;gBAE9C,GAAG,EAAE,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,GAAG,CAAC,EAAE,GAAG;IA8CpD,WAAW;IAyEjB,OAAO,CAAC,aAAa;YAUP,qBAAqB;IAU7B,kBAAkB,CAAC,SAAS,EAAE,GAAG;IAWvC,wBAAwB,CAAC,SAAS,EAAE,GAAG;IAWjC,yBAAyB;CAoJhC;AAED,eAAe,oBAAoB,CAAA"}