@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.
- package/.changeset/config.json +14 -0
- package/.github/copilot-instructions.md +39 -0
- package/.github/workflows/ci.yml +4 -1
- package/.github/workflows/manual-e2e.yml +6 -3
- package/.github/workflows/release.yml +64 -15
- package/.github/workflows/stale.yml +2 -4
- package/.husky/pre-push +15 -0
- package/CHANGELOG.md +126 -134
- package/MIGRATION.md +16 -6
- package/README.md +84 -3
- package/TODO.md +263 -0
- package/config.schema.json +229 -36
- package/dist/SwitchBotHAPPlatform.d.ts +133 -0
- package/dist/SwitchBotHAPPlatform.d.ts.map +1 -0
- package/dist/SwitchBotHAPPlatform.js +555 -0
- package/dist/SwitchBotHAPPlatform.js.map +1 -0
- package/dist/SwitchBotMatterPlatform.d.ts +141 -0
- package/dist/SwitchBotMatterPlatform.d.ts.map +1 -0
- package/dist/SwitchBotMatterPlatform.js +536 -0
- package/dist/SwitchBotMatterPlatform.js.map +1 -0
- package/dist/device-types.d.ts +31 -0
- package/dist/device-types.d.ts.map +1 -0
- package/dist/device-types.js +246 -0
- package/dist/device-types.js.map +1 -0
- package/dist/deviceCommandMapper.d.ts +10 -0
- package/dist/deviceCommandMapper.d.ts.map +1 -0
- package/dist/deviceCommandMapper.js +319 -0
- package/dist/deviceCommandMapper.js.map +1 -0
- package/dist/deviceFactory.d.ts +3 -2
- package/dist/deviceFactory.d.ts.map +1 -1
- package/dist/deviceFactory.js +107 -29
- package/dist/deviceFactory.js.map +1 -1
- package/dist/devices/genericDevice.d.ts +59 -37
- package/dist/devices/genericDevice.d.ts.map +1 -1
- package/dist/devices/genericDevice.js +376 -78
- package/dist/devices/genericDevice.js.map +1 -1
- package/dist/errors.d.ts +38 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +32 -0
- package/dist/errors.js.map +1 -0
- package/dist/homebridge-ui/device-types.js +246 -0
- package/dist/homebridge-ui/device-types.js.map +1 -0
- package/dist/homebridge-ui/deviceCommandMapper.js +319 -0
- package/dist/homebridge-ui/deviceCommandMapper.js.map +1 -0
- package/dist/homebridge-ui/endpoints/config.d.ts +3 -0
- package/dist/homebridge-ui/endpoints/config.d.ts.map +1 -0
- package/dist/homebridge-ui/endpoints/config.js +90 -0
- package/dist/homebridge-ui/endpoints/config.js.map +1 -0
- package/dist/homebridge-ui/endpoints/devices.d.ts +6 -0
- package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -0
- package/dist/homebridge-ui/endpoints/devices.js +144 -0
- package/dist/homebridge-ui/endpoints/devices.js.map +1 -0
- package/dist/homebridge-ui/endpoints/discovery.d.ts +7 -0
- package/dist/homebridge-ui/endpoints/discovery.d.ts.map +1 -0
- package/dist/homebridge-ui/endpoints/discovery.js +219 -0
- package/dist/homebridge-ui/endpoints/discovery.js.map +1 -0
- package/dist/homebridge-ui/errors.js +32 -0
- package/dist/homebridge-ui/errors.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/config.js +90 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/config.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js +144 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js +219 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/server.js +11 -0
- package/dist/homebridge-ui/homebridge-ui/server.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js +108 -0
- package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js +111 -0
- package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/utils/logger.js +17 -0
- package/dist/homebridge-ui/homebridge-ui/utils/logger.js.map +1 -0
- package/dist/homebridge-ui/public/css/styles.css +483 -0
- package/dist/homebridge-ui/public/index.html +197 -621
- package/dist/homebridge-ui/public/js/advanced-settings.d.ts +3 -0
- package/dist/homebridge-ui/public/js/advanced-settings.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/advanced-settings.js +95 -0
- package/dist/homebridge-ui/public/js/advanced-settings.js.map +1 -0
- package/dist/homebridge-ui/public/js/advanced-settings.ts +94 -0
- package/dist/homebridge-ui/public/js/api.d.ts +66 -0
- package/dist/homebridge-ui/public/js/api.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/api.js +295 -0
- package/dist/homebridge-ui/public/js/api.js.map +1 -0
- package/dist/homebridge-ui/public/js/api.ts +355 -0
- package/dist/homebridge-ui/public/js/app.d.ts +2 -0
- package/dist/homebridge-ui/public/js/app.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/app.js +3722 -0
- package/dist/homebridge-ui/public/js/app.js.map +7 -0
- package/dist/homebridge-ui/public/js/app.ts +22 -0
- package/dist/homebridge-ui/public/js/constants.d.ts +2 -0
- package/dist/homebridge-ui/public/js/constants.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/constants.js +2 -0
- package/dist/homebridge-ui/public/js/constants.js.map +1 -0
- package/dist/homebridge-ui/public/js/constants.ts +1 -0
- package/dist/homebridge-ui/public/js/credentials.d.ts +3 -0
- package/dist/homebridge-ui/public/js/credentials.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/credentials.js +99 -0
- package/dist/homebridge-ui/public/js/credentials.js.map +1 -0
- package/dist/homebridge-ui/public/js/credentials.ts +105 -0
- package/dist/homebridge-ui/public/js/devices-delete.d.ts +3 -0
- package/dist/homebridge-ui/public/js/devices-delete.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/devices-delete.js +199 -0
- package/dist/homebridge-ui/public/js/devices-delete.js.map +1 -0
- package/dist/homebridge-ui/public/js/devices-delete.ts +227 -0
- package/dist/homebridge-ui/public/js/devices.d.ts +9 -0
- package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/devices.js +98 -0
- package/dist/homebridge-ui/public/js/devices.js.map +1 -0
- package/dist/homebridge-ui/public/js/devices.ts +106 -0
- package/dist/homebridge-ui/public/js/discovery.d.ts +9 -0
- package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/discovery.js +1201 -0
- package/dist/homebridge-ui/public/js/discovery.js.map +1 -0
- package/dist/homebridge-ui/public/js/discovery.ts +1335 -0
- package/dist/homebridge-ui/public/js/logger.d.ts +7 -0
- package/dist/homebridge-ui/public/js/logger.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/logger.js +17 -0
- package/dist/homebridge-ui/public/js/logger.js.map +1 -0
- package/dist/homebridge-ui/public/js/logger.ts +17 -0
- package/dist/homebridge-ui/public/js/modal.d.ts +5 -0
- package/dist/homebridge-ui/public/js/modal.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/modal.js +35 -0
- package/dist/homebridge-ui/public/js/modal.js.map +1 -0
- package/dist/homebridge-ui/public/js/modal.ts +35 -0
- package/dist/homebridge-ui/public/js/modals.d.ts +15 -0
- package/dist/homebridge-ui/public/js/modals.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/modals.js +675 -0
- package/dist/homebridge-ui/public/js/modals.js.map +1 -0
- package/dist/homebridge-ui/public/js/modals.ts +765 -0
- package/dist/homebridge-ui/public/js/render.d.ts +71 -0
- package/dist/homebridge-ui/public/js/render.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/render.js +960 -0
- package/dist/homebridge-ui/public/js/render.js.map +1 -0
- package/dist/homebridge-ui/public/js/render.ts +1084 -0
- package/dist/homebridge-ui/public/js/toast.d.ts +6 -0
- package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/toast.js +38 -0
- package/dist/homebridge-ui/public/js/toast.js.map +1 -0
- package/dist/homebridge-ui/public/js/toast.ts +44 -0
- package/dist/homebridge-ui/public/js/types.d.ts +23 -0
- package/dist/homebridge-ui/public/js/types.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/types.js +2 -0
- package/dist/homebridge-ui/public/js/types.js.map +1 -0
- package/dist/homebridge-ui/public/js/types.ts +26 -0
- package/dist/homebridge-ui/server.d.ts +1 -3
- package/dist/homebridge-ui/server.d.ts.map +1 -1
- package/dist/homebridge-ui/server.js +8 -471
- package/dist/homebridge-ui/server.js.map +1 -1
- package/dist/homebridge-ui/settings.js +8 -0
- package/dist/homebridge-ui/settings.js.map +1 -0
- package/dist/homebridge-ui/switchbotClient.js +247 -0
- package/dist/homebridge-ui/switchbotClient.js.map +1 -0
- package/dist/homebridge-ui/utils/config-parser.d.ts +39 -0
- package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -0
- package/dist/homebridge-ui/utils/config-parser.js +108 -0
- package/dist/homebridge-ui/utils/config-parser.js.map +1 -0
- package/dist/homebridge-ui/utils/device-migration.d.ts +35 -0
- package/dist/homebridge-ui/utils/device-migration.d.ts.map +1 -0
- package/dist/homebridge-ui/utils/device-migration.js +111 -0
- package/dist/homebridge-ui/utils/device-migration.js.map +1 -0
- package/dist/homebridge-ui/utils/logger.d.ts +7 -0
- package/dist/homebridge-ui/utils/logger.d.ts.map +1 -0
- package/dist/homebridge-ui/utils/logger.js +17 -0
- package/dist/homebridge-ui/utils/logger.js.map +1 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +12 -2
- package/dist/index.js.map +1 -1
- package/dist/settings.d.ts +1 -0
- package/dist/settings.d.ts.map +1 -1
- package/dist/settings.js +1 -0
- package/dist/settings.js.map +1 -1
- package/dist/switchbotClient.d.ts +12 -10
- package/dist/switchbotClient.d.ts.map +1 -1
- package/dist/switchbotClient.js +156 -103
- package/dist/switchbotClient.js.map +1 -1
- package/dist/utils.d.ts +76 -1
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +1121 -4
- package/dist/utils.js.map +1 -1
- package/docs/assets/highlight.css +16 -2
- package/docs/assets/main.js +1 -1
- package/docs/index.html +82 -5
- package/docs/variables/default.html +3 -1
- package/eslint.config.js +9 -5
- package/nodemon.json +2 -2
- package/package.json +34 -21
- package/scripts/build-ui.js +37 -0
- package/scripts/free-dev-ports.mjs +105 -0
- package/scripts/generate-matter-maps.js +34 -17
- package/scripts/sync-device-types.mjs +31 -0
- package/src/SwitchBotHAPPlatform.ts +558 -0
- package/src/SwitchBotMatterPlatform.ts +538 -0
- package/src/device-types.js +246 -0
- package/src/device-types.js.map +1 -0
- package/src/device-types.ts +261 -0
- package/src/deviceCommandMapper.js +319 -0
- package/src/deviceCommandMapper.js.map +1 -0
- package/src/deviceCommandMapper.ts +333 -0
- package/src/deviceFactory.ts +125 -45
- package/src/devices/genericDevice.ts +411 -69
- package/src/errors.js +32 -0
- package/src/errors.js.map +1 -0
- package/src/errors.ts +35 -0
- package/src/homebridge-ui/endpoints/config.ts +110 -0
- package/src/homebridge-ui/endpoints/devices.ts +153 -0
- package/src/homebridge-ui/endpoints/discovery.ts +240 -0
- package/src/homebridge-ui/public/css/styles.css +483 -0
- package/src/homebridge-ui/public/index.html +197 -621
- package/src/homebridge-ui/public/js/advanced-settings.ts +94 -0
- package/src/homebridge-ui/public/js/api.ts +355 -0
- package/src/homebridge-ui/public/js/app.ts +22 -0
- package/src/homebridge-ui/public/js/constants.ts +1 -0
- package/src/homebridge-ui/public/js/credentials.ts +105 -0
- package/src/homebridge-ui/public/js/devices-delete.ts +227 -0
- package/src/homebridge-ui/public/js/devices.ts +106 -0
- package/src/homebridge-ui/public/js/discovery.ts +1335 -0
- package/src/homebridge-ui/public/js/logger.ts +17 -0
- package/src/homebridge-ui/public/js/modal.ts +35 -0
- package/src/homebridge-ui/public/js/modals.ts +765 -0
- package/src/homebridge-ui/public/js/render.ts +1084 -0
- package/src/homebridge-ui/public/js/toast.ts +44 -0
- package/src/homebridge-ui/public/js/types.ts +26 -0
- package/src/homebridge-ui/server.ts +9 -554
- package/src/homebridge-ui/utils/config-parser.ts +125 -0
- package/src/homebridge-ui/utils/device-migration.ts +144 -0
- package/src/homebridge-ui/utils/logger.ts +17 -0
- package/src/index.ts +12 -2
- package/src/settings.js +8 -0
- package/src/settings.js.map +1 -0
- package/src/settings.ts +2 -0
- package/src/switchbotClient.js +247 -0
- package/src/switchbotClient.js.map +1 -0
- package/src/switchbotClient.ts +177 -114
- package/src/utils.ts +1133 -5
- package/test/client/switchbot-client-debounce.spec.ts +35 -0
- package/test/client/switchbot-client-openapi.spec.ts +19 -0
- package/test/client/switchbotClient.spec.ts +64 -0
- package/test/device/device-mapping.spec.ts +23 -0
- package/test/device/deviceBase.spec.ts +26 -0
- package/test/device/deviceFactory-edge.spec.ts +15 -0
- package/test/device/deviceFactory.spec.ts +33 -0
- package/test/device/fan-swing.spec.ts +34 -0
- package/test/device/genericDevice-blepoll.spec.ts +47 -0
- package/test/device/irdevice.spec.ts +9 -0
- package/test/device/lock-users.spec.ts +35 -0
- package/test/device/matter-descriptors.spec.ts +22 -0
- package/test/device/matter-device-state.spec.ts +37 -0
- package/test/e2e/run-e2e.spec.ts +18 -19
- package/test/errors/errors.spec.ts +10 -0
- package/test/helpers/matter-harness.ts +20 -9
- package/test/homebridge-ui/server.spec.ts +9 -0
- package/test/platform/accessory-restore.spec.ts +37 -0
- package/test/platform/matter-childbridge.spec.ts +34 -0
- package/test/platform/matter-integration.spec.ts +33 -0
- package/test/platform/platform-edge.spec.ts +73 -0
- package/test/platform/platform.integration.spec.ts +34 -0
- package/test/utils/utils-extra.spec.ts +10 -0
- package/test/utils/utils.spec.ts +53 -0
- package/todo/TODO.md +80 -0
- package/tsconfig.ui.json +11 -0
- package/.github/npm-version-script-esm.js +0 -97
- package/.github/workflows/beta-release.yml +0 -52
- package/dist/platform.d.ts +0 -35
- package/dist/platform.d.ts.map +0 -1
- package/dist/platform.js +0 -945
- package/dist/platform.js.map +0 -1
- package/src/platform.ts +0 -963
- package/test/accessory-restore.spec.ts +0 -73
- package/test/device-mapping.spec.ts +0 -37
- package/test/deviceFactory.spec.ts +0 -18
- package/test/fan-swing.spec.ts +0 -29
- package/test/lock-users.spec.ts +0 -44
- package/test/matter-childbridge.spec.ts +0 -55
- package/test/matter-descriptors.spec.ts +0 -97
- package/test/matter-device-state.spec.ts +0 -101
- package/test/matter-integration.spec.ts +0 -70
- package/test/platform.integration.spec.ts +0 -55
- package/test/switchbot-client-debounce.spec.ts +0 -131
- package/test/switchbot-client-openapi.spec.ts +0 -56
- package/test/switchbotClient.spec.ts +0 -10
- 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
|
-
|
|
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
|
-
|
|
6
|
-
|
|
7
|
-
|
|
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
|
-
|
|
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
|
-
|
|
547
|
-
|
|
548
|
-
|
|
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
|