@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,44 @@
1
+ import './types.js'
2
+ import { uiLog } from './logger.js'
3
+
4
+ function showToast(
5
+ method: 'success' | 'error' | 'warning' | 'info',
6
+ message: string,
7
+ title = 'SwitchBot',
8
+ ): void {
9
+ try {
10
+ // Defensive: check for window and homebridge existence
11
+ const hb = typeof window !== 'undefined' ? (window as any).homebridge : undefined
12
+ const toast = hb && typeof hb.toast === 'object' ? hb.toast : undefined
13
+ const fn = toast && typeof toast[method] === 'function' ? toast[method] : undefined
14
+ if (fn) {
15
+ try {
16
+ fn(message, title)
17
+ return
18
+ } catch (err) {
19
+ uiLog.warn(`Toast ${method} threw:`, err)
20
+ }
21
+ }
22
+ // Fallback: log to console
23
+ uiLog.info(`[Toast:${method}] ${title} - ${message}`)
24
+ } catch (e) {
25
+ uiLog.warn(`Toast ${method} outer error:`, e)
26
+ uiLog.info(`[Toast:${method}] ${title} - ${message}`)
27
+ }
28
+ }
29
+
30
+ export function toastSuccess(message: string, title?: string): void {
31
+ showToast('success', message, title)
32
+ }
33
+
34
+ export function toastError(message: string, title?: string): void {
35
+ showToast('error', message, title)
36
+ }
37
+
38
+ export function toastWarning(message: string, title?: string): void {
39
+ showToast('warning', message, title)
40
+ }
41
+
42
+ export function toastInfo(message: string, title?: string): void {
43
+ showToast('info', message, title)
44
+ }
@@ -0,0 +1,26 @@
1
+ // Global homebridge API type declarations
2
+ declare global {
3
+ interface HomebridgeToastApi {
4
+ success?: (message: string, title?: string) => void
5
+ error?: (message: string, title?: string) => void
6
+ warning?: (message: string, title?: string) => void
7
+ info?: (message: string, title?: string) => void
8
+ }
9
+
10
+ interface HomebridgePluginUiAPI {
11
+ request: (endpoint: string, params?: any) => Promise<any>
12
+ getPluginConfig?: () => Promise<any[]>
13
+ updatePluginConfig?: (config: any[]) => Promise<void>
14
+ savePluginConfig?: () => Promise<void>
15
+ closeSettings?: () => void
16
+ showSpinner?: () => void
17
+ hideSpinner?: () => void
18
+ disableSaveButton?: () => void
19
+ enableSaveButton?: () => void
20
+ toast?: HomebridgeToastApi
21
+ }
22
+
23
+ let homebridge: HomebridgePluginUiAPI
24
+ }
25
+
26
+ export {}
@@ -1,531 +1,14 @@
1
- /* eslint-disable no-console */
2
- import { HomebridgePluginUiServer, RequestError } from '@homebridge/plugin-ui-utils'
3
- import fs from 'node:fs/promises'
1
+ import { HomebridgePluginUiServer } from '@homebridge/plugin-ui-utils'
4
2
 
5
- const server = new HomebridgePluginUiServer()
6
-
7
- function getDevicesRef(platform: any): any[] {
8
- if (!platform || typeof platform !== 'object') {
9
- return []
10
- }
11
-
12
- if (Array.isArray(platform.devices)) {
13
- return platform.devices
14
- }
15
-
16
- platform.devices = []
17
- return platform.devices
18
- }
19
-
20
- function getDeviceArrays(platform: any): any[][] {
21
- const rootDevices = getDevicesRef(platform)
22
- return [rootDevices]
23
- }
24
-
25
- function getAllDevices(platform: any): any[] {
26
- const all = getDevicesRef(platform)
27
- const seen = new Set<string>()
28
-
29
- return all.filter((d: any) => {
30
- const id = String(d?.deviceId ?? d?.id ?? '').trim().toLowerCase()
31
- if (!id || seen.has(id)) {
32
- return false
33
- }
34
- seen.add(id)
35
- return true
36
- })
37
- }
38
-
39
- function getCredential(platform: any, key: 'openApiToken' | 'openApiSecret') {
40
- return platform?.[key]
41
- }
42
-
43
- function setCredential(platform: any, key: 'openApiToken' | 'openApiSecret', value: string) {
44
- if (!platform || typeof platform !== 'object') {
45
- return
46
- }
47
-
48
- platform[key] = value
49
- }
50
-
51
- // Helper function to find the SwitchBot platform config
52
- async function getSwitchBotPlatformConfig() {
53
- const cfgPath = server.homebridgeConfigPath
54
- if (!cfgPath) {
55
- throw new Error('HOMEBRIDGE_CONFIG_PATH not set')
56
- }
57
-
58
- const raw = await fs.readFile(cfgPath, 'utf8')
59
- const cfg = JSON.parse(raw)
60
- const platforms = Array.isArray(cfg.platforms) ? cfg.platforms : []
61
-
62
- const candidates = platforms.filter((p: any) => {
63
- const platformName = p.platform || p.name || ''
64
- return !!platformName && /switchbot/i.test(String(platformName))
65
- })
66
-
67
- const platform = candidates.find((p: any) => String(p.platform || '').toLowerCase() === 'switchbot')
68
- ?? candidates[0]
69
-
70
- if (platform) {
71
- return { config: cfg, platform, cfgPath }
72
- }
73
-
74
- throw new Error('SwitchBot platform not found in config')
75
- }
76
-
77
- server.onRequest('/devices', async () => {
78
- try {
79
- const cfgPath = server.homebridgeConfigPath
80
- if (!cfgPath) {
81
- throw new Error('HOMEBRIDGE_CONFIG_PATH not set')
82
- }
83
-
84
- const raw = await fs.readFile(cfgPath, 'utf8')
85
- const cfg = JSON.parse(raw)
86
- const found: Array<{ id: string, name?: string, type?: string, connectionPreference?: string, room?: string }> = []
87
-
88
- const platforms = Array.isArray(cfg.platforms) ? cfg.platforms : []
89
- for (const p of platforms) {
90
- try {
91
- const platformName = p.platform || p.name || ''
92
- // Match known SwitchBot platform identifiers
93
- if (!platformName || !/switchbot/i.test(String(platformName))) {
94
- continue
95
- }
96
-
97
- const devices = getAllDevices(p)
98
- for (const d of devices) {
99
- const id = d.deviceId ?? d.id
100
- if (!id) {
101
- continue
102
- }
103
-
104
- found.push({
105
- id,
106
- name: d.configDeviceName ?? d.name ?? d.deviceName,
107
- type: d.configDeviceType ?? d.type ?? d.deviceType,
108
- connectionPreference: d.connectionPreference ?? d.connection ?? 'auto',
109
- room: d.room || d.location || undefined,
110
- })
111
- }
112
- } catch (e) {
113
- // ignore malformed platform entries
114
- }
115
- }
116
-
117
- console.log('[SwitchBot UI] GET /devices - Found', found.length, 'devices in', cfgPath)
118
- return { success: true, data: found }
119
- } catch (e) {
120
- console.error('[SwitchBot UI] Error in /devices:', e)
121
- throw new RequestError('Failed to read Homebridge config', e)
122
- }
123
- })
124
-
125
- server.onRequest('/discover', async () => {
126
- try {
127
- const { platform } = await getSwitchBotPlatformConfig()
128
-
129
- console.log('[SwitchBot UI] GET /discover - Platform config:', Object.keys(platform))
130
-
131
- const token = getCredential(platform, 'openApiToken') || platform.token
132
- if (!token) {
133
- console.error('[SwitchBot UI] No API token found in platform config')
134
- throw new Error('OpenAPI token not configured. Please save your credentials first.')
135
- }
136
-
137
- console.log('[SwitchBot UI] GET /discover - Fetching devices from SwitchBot API')
138
-
139
- const url = 'https://api.switch-bot.com/v1.0/devices'
140
- const opts = {
141
- headers: {
142
- Authorization: token,
143
- },
144
- }
145
-
146
- console.log('[SwitchBot UI] GET /discover - Making request to', url)
147
- const resp = await fetch(url, opts)
148
-
149
- console.log('[SwitchBot UI] GET /discover - API response status:', resp.status)
150
-
151
- if (!resp.ok) {
152
- const errorText = await resp.text()
153
- console.error('[SwitchBot UI] API error response:', errorText)
154
- throw new Error(`SwitchBot API returned ${resp.status}: ${errorText}`)
155
- }
156
-
157
- const data = await resp.json()
158
- console.log('[SwitchBot UI] GET /discover - Full API response:', JSON.stringify(data, null, 2))
159
-
160
- // Handle different response formats
161
- let devices: any[] = []
162
- if (Array.isArray(data)) {
163
- devices = data
164
- } else if (Array.isArray(data.body)) {
165
- devices = data.body
166
- } else if (Array.isArray(data.devices)) {
167
- devices = data.devices
168
- } else if (data.body && typeof data.body === 'object') {
169
- // Sometimes the response has devices listed within body
170
- const bodyKeys = Object.keys(data.body)
171
- if (bodyKeys.includes('deviceList')) {
172
- devices = Array.isArray(data.body.deviceList) ? data.body.deviceList : []
173
- } else if (bodyKeys.includes('devices')) {
174
- devices = Array.isArray(data.body.devices) ? data.body.devices : []
175
- }
176
- }
177
-
178
- if (!Array.isArray(devices)) {
179
- console.warn('[SwitchBot UI] Could not find devices array in API response. Response keys:', Object.keys(data), 'Body keys:', data.body ? Object.keys(data.body) : 'N/A')
180
- devices = []
181
- }
182
-
183
- const discovered = devices.map((d: any) => ({
184
- id: d.deviceId || d.id,
185
- name: d.deviceName || d.name,
186
- type: d.deviceType || d.type || 'unknown',
187
- model: d.deviceModel || d.model,
188
- enabled: d.enableCloudService !== false,
189
- }))
190
-
191
- console.log('[SwitchBot UI] GET /discover - Found', discovered.length, 'devices from API')
192
- return { success: true, data: discovered }
193
- } catch (e) {
194
- console.error('[SwitchBot UI] Error in /discover:', e instanceof Error ? e.message : String(e))
195
- throw new RequestError(`Failed to discover devices: ${e instanceof Error ? e.message : String(e)}`, e)
196
- }
197
- })
198
-
199
- server.onRequest('/add-device', async (body: any) => {
200
- try {
201
- const { deviceId, name, type } = body
202
-
203
- if (!deviceId) {
204
- throw new Error('Device ID is required')
205
- }
206
-
207
- const { config, platform, cfgPath } = await getSwitchBotPlatformConfig()
208
-
209
- console.log('[SwitchBot UI] POST /add-device - Adding device:', deviceId, 'to', cfgPath)
210
- const deviceArrays = getDeviceArrays(platform)
211
- const allDevices = getAllDevices(platform)
212
-
213
- // Check if already exists
214
- const normalizedDeviceId = String(deviceId).trim().toLowerCase()
215
- const exists = allDevices.some((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)
216
- if (exists) {
217
- console.log('[SwitchBot UI] Device already exists:', deviceId)
218
- return {
219
- success: true,
220
- data: {
221
- message: 'Device already in config',
222
- alreadyExists: true,
223
- },
224
- }
225
- }
226
-
227
- // Add device
228
- let addCount = 0
229
- for (const ref of deviceArrays) {
230
- const refHasDevice = ref.some((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)
231
- if (!refHasDevice) {
232
- const newDevice = {
233
- deviceId,
234
- configDeviceName: name,
235
- configDeviceType: type,
236
- }
237
- ref.push(newDevice)
238
- console.log('[SwitchBot UI] Added device to config:', JSON.stringify(newDevice))
239
- addCount++
240
- }
241
- }
242
-
243
- // Save config
244
- const configJson = JSON.stringify(config, null, 2)
245
- console.log(`[SwitchBot UI] Writing config to file (size: ${configJson.length} bytes)`)
246
- await fs.writeFile(cfgPath, configJson, 'utf8')
247
-
248
- console.log('[SwitchBot UI] Device added successfully:', deviceId, 'saved to', cfgPath)
249
-
250
- return {
251
- success: true,
252
- data: {
253
- message: `Device "${name}" added successfully`,
254
- alreadyExists: false,
255
- added: addCount > 0,
256
- },
257
- }
258
- } catch (e) {
259
- console.error('[SwitchBot UI] Error in /add-device:', e)
260
- throw new RequestError('Failed to add device', e)
261
- }
262
- })
263
-
264
- server.onRequest('/add-devices', async (body: any) => {
265
- try {
266
- const { devices } = body
267
-
268
- if (!Array.isArray(devices) || devices.length === 0) {
269
- throw new Error('Devices array is required and must not be empty')
270
- }
3
+ import { registerConfigEndpoints } from './endpoints/config.js'
4
+ import { registerDeviceEndpoints } from './endpoints/devices.js'
5
+ import { registerDiscoveryEndpoint } from './endpoints/discovery.js'
271
6
 
272
- const { config, platform, cfgPath } = await getSwitchBotPlatformConfig()
273
-
274
- console.log('[SwitchBot UI] POST /add-devices - Adding', devices.length, 'device(s) in bulk to', cfgPath)
275
- const deviceArrays = getDeviceArrays(platform)
276
- const allDevices = getAllDevices(platform)
277
-
278
- let addedCount = 0
279
- let skippedCount = 0
280
- const results: Array<Record<string, any>> = []
281
-
282
- for (const { deviceId, name, type } of devices) {
283
- if (!deviceId) {
284
- results.push({ deviceId: '?', success: false, message: 'Device ID is required' })
285
- skippedCount++
286
- continue
287
- }
288
-
289
- const normalizedDeviceId = String(deviceId).trim().toLowerCase()
290
- const exists = allDevices.some((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)
291
-
292
- if (exists) {
293
- console.log('[SwitchBot UI] Device already exists:', deviceId)
294
- results.push({ deviceId, success: true, message: 'Already in config', alreadyExists: true })
295
- skippedCount++
296
- continue
297
- }
298
-
299
- // Add device
300
- let addedToArray = false
301
- for (const ref of deviceArrays) {
302
- const refHasDevice = ref.some((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)
303
- if (!refHasDevice) {
304
- const newDevice = {
305
- deviceId,
306
- configDeviceName: name,
307
- configDeviceType: type,
308
- }
309
- ref.push(newDevice)
310
- addedToArray = true
311
- }
312
- }
313
-
314
- if (addedToArray) {
315
- console.log('[SwitchBot UI] Added device in bulk:', deviceId)
316
- results.push({ deviceId, success: true, message: `Device "${name}" added` })
317
- addedCount++
318
- } else {
319
- results.push({ deviceId, success: false, message: 'Failed to add device' })
320
- skippedCount++
321
- }
322
- }
323
-
324
- // Save config once for all devices
325
- const configJson = JSON.stringify(config, null, 2)
326
- console.log(`[SwitchBot UI] Writing bulk config to file (size: ${configJson.length} bytes, devices: ${addedCount})`)
327
- await fs.writeFile(cfgPath, configJson, 'utf8')
328
-
329
- console.log('[SwitchBot UI] Bulk device add completed: Added', addedCount, 'Skipped', skippedCount)
330
-
331
- return {
332
- success: true,
333
- data: {
334
- message: `Added ${addedCount} device(s)${skippedCount > 0 ? ` (${skippedCount} skipped)` : ''}`,
335
- addedCount,
336
- skippedCount,
337
- results,
338
- },
339
- }
340
- } catch (e) {
341
- console.error('[SwitchBot UI] Error in /add-devices:', e)
342
- throw new RequestError(`Failed to add devices: ${e instanceof Error ? e.message : String(e)}`, e)
343
- }
344
- })
345
-
346
- server.onRequest('/update-device', async (body: any) => {
347
- try {
348
- const { deviceId, configDeviceName, configDeviceType, connectionPreference, room } = body
349
-
350
- console.log('[SwitchBot UI] POST /update-device - Full request body:', JSON.stringify(body, null, 2))
351
-
352
- if (!deviceId) {
353
- throw new Error('Device ID is required')
354
- }
355
-
356
- const { config, platform, cfgPath } = await getSwitchBotPlatformConfig()
357
-
358
- console.log('[SwitchBot UI] POST /update-device - Updating device:', deviceId)
359
- console.log('[SwitchBot UI] Parameters - name:', configDeviceName, 'type:', configDeviceType, 'pref:', connectionPreference, 'room:', room)
360
-
361
- const deviceArrays = getDeviceArrays(platform)
362
- const devices = getAllDevices(platform)
363
- const normalizedDeviceId = String(deviceId).trim().toLowerCase()
364
- const device = devices.find((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)
365
- if (!device) {
366
- console.error('[SwitchBot UI] Device not found. Available devices:', devices.map((d: any) => d.deviceId ?? d.id))
367
- throw new Error('Device not found in config')
368
- }
369
-
370
- console.log('[SwitchBot UI] Found device before update:', JSON.stringify(device, null, 2))
371
-
372
- // Update properties in all config branches
373
- for (const ref of deviceArrays) {
374
- for (const entry of ref) {
375
- const entryId = String(entry.deviceId ?? entry.id ?? '').trim().toLowerCase()
376
- if (entryId !== normalizedDeviceId) {
377
- continue
378
- }
379
-
380
- if (configDeviceName !== undefined && configDeviceName !== null && String(configDeviceName).trim() !== '') {
381
- entry.configDeviceName = configDeviceName
382
- }
383
- if (configDeviceType !== undefined && configDeviceType !== null && String(configDeviceType).trim() !== '') {
384
- entry.configDeviceType = configDeviceType
385
- }
386
- if (connectionPreference !== undefined) {
387
- entry.connectionPreference = connectionPreference
388
- }
389
- if (room !== undefined) {
390
- entry.room = room || undefined
391
- }
392
- }
393
- }
394
-
395
- if (configDeviceName !== undefined && configDeviceName !== null && String(configDeviceName).trim() !== '') {
396
- console.log('[SwitchBot UI] ✓ Updated configDeviceName to:', configDeviceName)
397
- }
398
- if (configDeviceType !== undefined && configDeviceType !== null && String(configDeviceType).trim() !== '') {
399
- console.log('[SwitchBot UI] ✓ Updated configDeviceType to:', configDeviceType)
400
- }
401
- if (connectionPreference !== undefined) {
402
- console.log('[SwitchBot UI] ✓ Updated connectionPreference to:', connectionPreference)
403
- }
404
- if (room !== undefined) {
405
- console.log('[SwitchBot UI] ✓ Updated room to:', room)
406
- }
407
-
408
- console.log('[SwitchBot UI] Device after updates:', JSON.stringify(device, null, 2))
409
-
410
- // Save config
411
- await fs.writeFile(cfgPath, JSON.stringify(config, null, 2), 'utf8')
412
- console.log('[SwitchBot UI] ✓ Config file saved successfully:', cfgPath)
413
-
414
- console.log('[SwitchBot UI] Device updated successfully:', deviceId)
415
-
416
- return {
417
- success: true,
418
- data: { message: `Device updated successfully - type: ${configDeviceType || device.configDeviceType}` },
419
- }
420
- } catch (e) {
421
- console.error('[SwitchBot UI] Error in /update-device:', e instanceof Error ? e.message : String(e), e)
422
- throw new RequestError(`Failed to update device: ${e instanceof Error ? e.message : String(e)}`, e)
423
- }
424
- })
425
-
426
- server.onRequest('/delete-device', async (body: any) => {
427
- try {
428
- const { deviceId } = body
429
-
430
- if (!deviceId) {
431
- throw new Error('Device ID is required')
432
- }
433
-
434
- const { config, platform, cfgPath } = await getSwitchBotPlatformConfig()
435
-
436
- console.log('[SwitchBot UI] POST /delete-device - Deleting:', deviceId)
437
-
438
- const deviceArrays = getDeviceArrays(platform)
439
- const normalizedDeviceId = String(deviceId).trim().toLowerCase()
440
- const allDevices = getAllDevices(platform)
441
- const existing = allDevices.find((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)
442
- if (!existing) {
443
- throw new Error('Device not found in config')
444
- }
445
-
446
- const deviceName = existing.configDeviceName || deviceId
447
- let removedCount = 0
448
- for (const ref of deviceArrays) {
449
- for (let i = ref.length - 1; i >= 0; i--) {
450
- const id = String(ref[i]?.deviceId ?? ref[i]?.id ?? '').trim().toLowerCase()
451
- if (id === normalizedDeviceId) {
452
- ref.splice(i, 1)
453
- removedCount++
454
- }
455
- }
456
- }
457
-
458
- if (!removedCount) {
459
- throw new Error('Device not found in config')
460
- }
461
-
462
- // Save config
463
- await fs.writeFile(cfgPath, JSON.stringify(config, null, 2), 'utf8')
464
-
465
- console.log('[SwitchBot UI] Device deleted successfully:', deviceId, 'saved to', cfgPath)
466
-
467
- return {
468
- success: true,
469
- data: { message: `Device "${deviceName}" removed from config` },
470
- }
471
- } catch (e) {
472
- console.error('[SwitchBot UI] Error in /delete-device:', e)
473
- throw new RequestError('Failed to delete device', e)
474
- }
475
- })
476
-
477
- server.onRequest('/credentials', async (body: any) => {
478
- try {
479
- // Handle both GET and POST requests
480
- if (!body || Object.keys(body).length === 0) {
481
- // GET request - return current status
482
- const { platform } = await getSwitchBotPlatformConfig()
483
-
484
- const token = getCredential(platform, 'openApiToken')
485
- const secret = getCredential(platform, 'openApiSecret')
486
- const status = {
487
- hasToken: !!token,
488
- hasSecret: !!secret,
489
- tokenLength: token ? String(token).length : 0,
490
- secretLength: secret ? String(secret).length : 0,
491
- }
492
-
493
- console.log('[SwitchBot UI] GET /credentials - Status:', status)
494
- return { success: true, data: status }
495
- } else {
496
- // POST request - save credentials
497
- const { token, secret } = body
498
-
499
- if (!token || !secret) {
500
- throw new Error('Token and secret are required')
501
- }
502
-
503
- const { config, platform, cfgPath } = await getSwitchBotPlatformConfig()
504
-
505
- console.log('[SwitchBot UI] POST /credentials - Saving to platform:', platform.platform || platform.name)
506
- console.log('[SwitchBot UI] Config path:', cfgPath)
507
- console.log('[SwitchBot UI] Token length:', token.length, 'Secret length:', secret.length)
508
-
509
- // Save token and secret on the same config branch used by this platform
510
- setCredential(platform, 'openApiToken', token)
511
- setCredential(platform, 'openApiSecret', secret)
512
-
513
- // Write back to config file
514
- await fs.writeFile(cfgPath, JSON.stringify(config, null, 2), 'utf8')
515
-
516
- console.log('[SwitchBot UI] Credentials saved successfully')
7
+ const server = new HomebridgePluginUiServer()
517
8
 
518
- return {
519
- success: true,
520
- data: { message: 'Credentials saved successfully' },
521
- }
522
- }
523
- } catch (e) {
524
- console.error('[SwitchBot UI] Error in /credentials:', e)
525
- throw new RequestError('Failed to handle credentials request', e)
526
- }
527
- })
9
+ // Register all endpoints
10
+ registerConfigEndpoints(server)
11
+ registerDiscoveryEndpoint(server)
12
+ registerDeviceEndpoints(server)
528
13
 
529
14
  server.ready()
530
-
531
- export default server