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